A \<new line> is not the correct way to get a <new line> in the history.
Memory
Lets deal only with history lines as they are kept in shell memory (not disk).
Lets type a couple of commands as you did:
$ echo -n "this is a test for a ";\ > echo "two line command"
What was stored in memory as the line just written?
$ history 2 514 echo -n "this is a test for a ";echo "two line command" 515 history 2
As you can see, the "line continuation", a backslash followed by a newline, was removed.
As it should (from man bash):
If a \ pair appears, and the backslash is not itself quoted, the \ is treated as a line continuation (that is, it is removed from the input stream and effectively ignored).
We may get a newline if we quote it:
$ echo " A test of > a new line" A test of a new line
And, at this point, the history will reflect that:
$ history 2 518 echo "A test of a new line" 519 history 2
A true multi-line command:
One possible example of a multi-line command is:
$ for a in one two > do echo "test $a" > done test one test two
Which will be collapsed into one history line if cmdhist is set:
$ shopt -p cmdhist lithist shopt -s cmdhist shopt -u lithist $ history 3 24 for a in one two; do echo "test $a"; done 25 shopt -p cmdhist lithist 26 history 3
The numbers for each command changed because at some point I cleared the history (in memory) with a history -c.
If you unset the cmdhist, you will get this:
$ shopt -u cmdhist $ for a in one two > do echo "test $a" > done test one test two $ history 5 5 shopt -u cmdhist 6 for a in one two 7 do echo "test $a" 8 done 9 history 5
Each line (not a full command) will be at a separate line in the history.
Even if the lithist is set:
$ shopt -s lithist $ for a in one two > do echo "test $a" > done test one test two $ history 5 12 shopt -s lithist 13 for a in one two 14 do echo "test $a" 15 done 16 history 5
But if both are set:
$ shopt -s cmdhist lithist $ for a in one two > do echo "test $a" > done $ history 5 23 history 15 24 shopt -p cmdhist lithist 25 shopt -s cmdhist lithist 26 for a in one two do echo "test $a" done 27 history 5
The for command was stored as a multiline command with "newlines" instead of semicolons (;). Compare with above where lithist wasn't set.
Disk
All the above was explained using the list of commands kept in the memory of the shell. No commands were written to the disk. The (default) file ~/.bash_history was not changed.
That file will be changed when the running shell exits. At that point in time the history will overwrite the file (if histappend isn't set), or will be appended otherwise.
If you want the commands to be committed to disk you need to have this set:
export PROMPT_COMMAND='history -a'
That will make each command line to be appended to file on each new command line.
Now, lets get down to business with cmdhist and lithist. It is not so simple as it may seem. But don't worry, all will be clear in a moment.
Let's say that you take the time to type all the commands below (there is no shortcut, no alias, no function, you need the actual commands, sorry).
To first clear all history in memory (history -c) and in disk (make a backup) (history -w) and then to try three times:
- With the default values of cmdhist (set) and lithist (unset).
- With both set
- With both un-set
- Setting lithist with an unset cmdhist makes no sense (you can test it).
List of commands to execute:
$ history -c ; history -w # Clear all history ( Please backup). $ shopt -s cmdhist; shopt -u lithist $ for a in one two > do echo "test $a" > done $ shopt -s cmdhist; shopt -s lithist $ for a in one two > do echo "test $a" > done $ shopt -u cmdhist; shopt -u lithist $ for a in one two > do echo "test $a" > done
You will end with this (in memory):
$ history 1 shopt -s cmdhist; shopt -u lithist 2 for a in one two; do echo "test $a"; done 3 shopt -s cmdhist; shopt -s lithist 4 for a in one two do echo "test $a" done 5 shopt -u cmdhist; shopt -u lithist 6 for a in one two 7 do echo "test $a" 8 done 9 history
The three multiline commands end as follows:
- one in line numbered 2 (one single line, one command).
- one in a multiline numbered 4 (one command in several lines)
- one in several lines numbered from 6 to 8
Ok, but what happen in file? say it alredy ....
finally, in file:
Simple, write to file and cat it to see this:
$ history -w ; cat "$HISTFILE" shopt -s cmdhist; shopt -u lithist for a in one two; do echo "test $a"; done shopt -s cmdhist; shopt -s lithist for a in one two do echo "test $a" done shopt -u cmdhist; shopt -u lithist for a in one two do echo "test $a" done history history -w ; cat "$HISTFILE"
No line numbers, only commands, there is no way to tell where a multiline starts and where it ends. There is no way to tell even if there is a multiline.
In fact, that is exactly what happens, if the commands are written to file as above, when the file is read back, any information about multilines gets lost.
There is only one delimiter (the newline), each lines is read back as one command.
Is there a solution to this, yes, to use an additional delimter.
The HISTTIMEFORMAT kind of does that.
HISTTIMEFORMAT
When this variable is set to some value, the time at which each command was executed gets stored in file as the seconds since epoch (yes, always seconds) after a comment (#) character.
If we set the variable and re-write the ~/.bash_history file, we get this:
$ HISTTIMEFORMAT='%F' $ history -w ; cat "$HISTFILE" #1490321397 shopt -s cmdhist; shopt -u lithist #1490321397 for a in one two; do echo "test $a"; done #1490321406 shopt -s cmdhist; shopt -s lithist #1490321406 for a in one two do echo "test $a" done #1490321418 shopt -u cmdhist; shopt -u lithist #1490321418 for a in one two #1490321418 do echo "test $a" #1490321420 done #1490321429 history #1490321439 history -w ; cat "$HISTFILE" #1490321530 HISTTIMEFORMAT='%FT%T ' #1490321571 history -w ; cat "$HISTFILE"
Now you can tell where and which line is a multiline.
The format '%FT%T ' shows the time but only when using the history command:
$ history 1 2017-03-23T22:09:57 shopt -s cmdhist; shopt -u lithist 2 2017-03-23T22:09:57 for a in one two; do echo "test $a"; done 3 2017-03-23T22:10:06 shopt -s cmdhist; shopt -s lithist 4 2017-03-23T22:10:06 for a in one two do echo "test $a" done 5 2017-03-23T22:10:18 shopt -u cmdhist; shopt -u lithist 6 2017-03-23T22:10:18 for a in one two 7 2017-03-23T22:10:18 do echo "test $a" 8 2017-03-23T22:10:20 done 9 2017-03-23T22:10:29 history 10 2017-03-23T22:10:39 history -w ; cat "$HISTFILE" 11 2017-03-23T22:12:10 HISTTIMEFORMAT='%F' 12 2017-03-23T22:12:51 history -w ; cat "$HISTFILE" 13 2017-03-23T22:15:30 history 14 2017-03-23T22:16:29 HISTTIMEFORMAT='%FT%T' 15 2017-03-23T22:16:31 history 16 2017-03-23T22:16:35 HISTTIMEFORMAT='%FT%T ' 17 2017-03-23T22:16:37 history