What I get:
host:~ user$ cat example.txt some texthost:~ stas$
What I want to get:
host:~ user$ cat example.txt some text host:~ stas$
Is there a way I can make cat behave like this?
I’m using bash on Mac OS X.
Answers:
Thank you for visiting the Q&A section on Magenaut. Please note that all the answers may not help you solve the issue immediately. So please treat them as advisements. If you found the post helpful (or not), leave a comment & I’ll get back to you as soon as possible.
Method 1
Most unix tools are designed to work well with text files. A text file consists of a sequence of lines. A line consists of a sequence of printable characters ending with a newline character. In particular, the last character of a non-empty text file is always a newline character. Evidently, example.txt contains only some text with no final newline, so it is not a text file.
cat does a simple job; turning arbitrary files into text files isn’t part of that job. Some other tools always turn their input into text files; if you aren’t sure the file you’re displaying ends with a newline, try running awk 1 instead of cat.
You can make the bash display its prompt on the next line if the previous command left the cursor somewhere other than the last margin. Put this in your .bashrc (variation by GetFree of a proposal by Dennis Williamson):
shopt -s promptvars PS1='$(printf "%$((COLUMNS-1))sr")'$PS1
Method 2
I prefer the following method…
cat example.txt ; echo
This doesn’t doesn’t evaluate the contents of example.txt or occasionally add a newline. It just echos a newline once the cat is done, is easy to remember, and no one is thinking about whether they’re using strong or weak quoting correctly.
The only downside, really, is that you’ll get an extra newline if the file has its own trailing newline.
Method 3
I started using @Gilles’s answer, but found that if the terminal changed the number of columns the prompt would no longer be at the start of a line as expected. This can happen for a variety of reasons, including tmux/screen splits, manual resizing of a GUI container, font changes, etc.
What I really wanted was something that would add a newline if the terminal would start printing its prompt at something other than the first column. To do this I needed to figure out how to get the current column, which I used this answer to get. The final working prompt configuration is below:
###
# Configure PS1 by using the old value but ensuring it starts on a new line.
###
__configure_prompt() {
PS1=""
if [ "$(__get_terminal_column)" != 0 ]; then
PS1="n"
fi
PS1+="$PS1_WITHOUT_PREPENDED_NEWLINE"
}
###
# Get the current terminal column value.
#
# From https://stackoverflow.com/a/2575525/549363.
###
__get_terminal_column() {
exec < /dev/tty
local oldstty=$(stty -g)
stty raw -echo min 0
echo -en "33[6n" > /dev/tty
local pos
IFS=';' read -r -d R -a pos
stty $oldstty
echo "$((${pos[1]} - 1))"
}
# Save the current PS1 for later.
PS1_WITHOUT_PREPENDED_NEWLINE="$PS1"
# Use our prompt configuration function, preserving whatever existing
# PROMPT_COMMAND might be configured.
PROMPT_COMMAND="__configure_prompt;$PROMPT_COMMAND"
Method 4
The problem with that could be that your example.txt does not have a newline at the end of your file.
Method 5
If you insist on using cat, this works for both types of files, with and without a newline at the end:
echo "`cat example.txt`"
You can turn it into a function with a name of your choice (even cat) in your .bashrc:
cat1(){ echo "`/bin/cat <a href="https://getridbug.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="664226">[email protected]</a>`";}
Method 6
you can add to .bashrc as well
PROMPT_COMMAND="printf 'n';$PROMPT_COMMAND"
works for me.
Method 7
If you use a modern Bash + terminal, you can use this solution based on a few similar answers:
# Get the "raw" form of the row and column
IFS=';' read -d R -p "$(tput u7)" -r -s row column
if [[ "$column" -ne 1 ]]
then
# Remove cruft from row to get the integer value
row="${row#*[}"
# Decrement column to avoid adding an empty space before the EOT marker
((column--))
# Add an inverted ␄ symbol and newline before the rest of $PS1
PS1="$(tput cup "$row" "$column")$(tput rev)␄$(tput sgr0)n${PS1}"
fi
Pros:
- Uses
tputrather than escape sequences for easy searching and (probably?) wider support. - Prints the Unicode end-of-transmission character (EOT) character (I couldn’t find a more appropriate EOF marker, if it exists) in inverted colours followed by a newline to indicate the missing newline at EOF. This has the advantage of being easy to spot, while being short (a single character). Outputting a newline before the rest of
$PS1ensures that it’s not clobbered by$PS1if that contains a carriage return character. - Easy to change to use other formatting (colours, bold, etc) and another marker, if you want to.
Cons:
- Uses
tputrather than escape sequences, so it might not work on older terminals.
All methods was sourced from stackoverflow.com or stackexchange.com, is licensed under cc by-sa 2.5, cc by-sa 3.0 and cc by-sa 4.0