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
A German immigrant ancestor has a "Registration Affidavit of Alien Enemy" on file. What does that mean exactly?
If gravity precedes the formation of a solar system, where did the mass come from that caused the gravity?
Assertions In A Mock Callout Test
How to leave only the following strings?
How is it possible to implement unitary operator when its size is exponential in inputs?
What is the evidence that custom checks in Northern Ireland are going to result in violence?
Why did Israel vote against lifting the American embargo on Cuba?
How to keep bees out of canned beverages?
Why aren't road bike wheels tiny?
Will the Antimagic Field spell cause elementals not summoned by magic to dissipate?
When speaking, how do you change your mind mid-sentence?
How to make an animal which can only breed for a certain number of generations?
/bin/ls sorts differently than just ls
Married in secret, can marital status in passport be changed at a later date?
What helicopter has the most rotor blades?
Compiling and throwing simple dynamic exceptions at runtime for JVM
How to mute a string and play another at the same time
How to deal when you're an actual imposter in academia?
A journey... into the MIND
What is the definining line between a helicopter and a drone a person can ride in?
Is Bran literally the world's memory?
Marquee sign letters
“Since the train was delayed for more than an hour, passengers were given a full refund.” – Why is there no article before “passengers”?
How to calculate density of unknown planet?
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;
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
add a comment |
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
10
How aboutpython 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
add a comment |
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
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
linux command-line redirection stdout
edited Apr 4 at 19:01
Peter Cordes
2,5451622
2,5451622
asked Apr 3 at 23:31
interstarinterstar
438613
438613
10
How aboutpython 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
add a comment |
10
How aboutpython 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
add a comment |
5 Answers
5
active
oldest
votes
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()
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 useprint
'sflush
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 thestdout
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
|
show 2 more comments
Run python with the unbuffered flag:
python -u myprog.py > output.txt
Output will then print in real time.
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
add a comment |
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.
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 usetail -F
or even better the follow function ofless
. But in all cases theflush
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
add a comment |
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'
.
add a comment |
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
add a comment |
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
);
);
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
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
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()
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 useprint
'sflush
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 thestdout
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
|
show 2 more comments
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()
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 useprint
'sflush
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 thestdout
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
|
show 2 more comments
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()
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()
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 useprint
'sflush
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 thestdout
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
|
show 2 more comments
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 useprint
'sflush
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 thestdout
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
|
show 2 more comments
Run python with the unbuffered flag:
python -u myprog.py > output.txt
Output will then print in real time.
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
add a comment |
Run python with the unbuffered flag:
python -u myprog.py > output.txt
Output will then print in real time.
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
add a comment |
Run python with the unbuffered flag:
python -u myprog.py > output.txt
Output will then print in real time.
Run python with the unbuffered flag:
python -u myprog.py > output.txt
Output will then print in real time.
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
add a comment |
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
add a comment |
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.
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 usetail -F
or even better the follow function ofless
. But in all cases theflush
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
add a comment |
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.
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 usetail -F
or even better the follow function ofless
. But in all cases theflush
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
add a comment |
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.
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.
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 usetail -F
or even better the follow function ofless
. But in all cases theflush
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
add a comment |
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 usetail -F
or even better the follow function ofless
. But in all cases theflush
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
add a comment |
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'
.
add a comment |
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'
.
add a comment |
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'
.
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'
.
answered Apr 4 at 19:18
Peter CordesPeter Cordes
2,5451622
2,5451622
add a comment |
add a comment |
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
add a comment |
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
add a comment |
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
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
edited Apr 8 at 22:57
Alfabravo
528413
528413
answered Apr 8 at 20:29
d-nd-n
11
11
add a comment |
add a comment |
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.
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
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
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
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
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