Let\'s say I have a file called original.txt
with this content:
red
blue
water
food
tree
This might work for you (GNU sed):
sed '/blue/,/gray/!b;//!d;/blue/r file2' file1
For the range of lines between blue
and gray
, delete lines that do not match either the first or the last lines in the range and read the lines to insert before the last line of the range.
EDIT:
The first sed command /blue/,/grey/!b
matches a range of lines i.e. the lines that range between a line containing blue
upto and including a line containing gray
. The !b
part means if the lines are not in that range then break out of the sed commands i.e. do not do any further sed processing for these lines, just print as normal.
The sed commands following will only affect those lines that are in the range between blue
and gray
.
The second command //!d
means: delete those lines that do not match either the start/end of the range i.e. blue
or gray
. The //
uses the regexp from a previous /.../
command. N.B. a delete command terminates any further sed processing for that line.
The sed commands following will only affect lines that containing either blue
or gray
.
The third command matches a line containing blue
and reads in lines from file2.
N.B. the lines containing blue
and grey
are processed by sed naturally and printed before the next line is read into the pattern space as are the lines not between blue
and gray
.
An alternative:
sed '/blue/,/gray/!b;//!d;/gray/e cat file2' file1
And another:
sed -ne '/blue/{p;r file2' -e ':a;n;/gray/!ba};p' file1
I see you asked for sed help, but ex handles this use-case in a bit more straightforward way. This may work for you (in a shell script):
ex original.txt << EOF_EX
/blue/+1,/gray/-1 d
r new.txt
w edited.txt
q!
EOF_EX
The above opens original.txt for editing, deletes the range of lines one past blue through one before gray, reads the file new.txt, and writes out file edited.txt.
Result:
$ cat edited.txt
red
blue
gray
green
black
yellow
purple
white
Alternatively you could replace "w edited.txt" with "wq" and it would save the changes in the original.txt file.
Also, notice the +1 and -1 syntax in the range, which can be very handy. For example if you had wanted to also remove the range-bounding patterns, "blue" and "gray" you could remove the +1 and -1 and it would mean delete the range, including the range-boundary patterns.
sed is for s/old/new/, that is all. For anything else you should be using awk:
$ awk 'NR==FNR{new = new $0 ORS; next} /gray/{f=0} !f{print} /blue/{printf "%s",new; f=1}' new.txt original.txt
red
blue
green
black
yellow
purple
gray
white
The above will work in any awk in any shell on any UNIX box and does not require any of the start/end regexps to be repeated. Very importantly it can also trivially be built upon to do anything else you want now or in future! For example to be able to specify the beginning and ending regexps as arguments would be:
$ awk -v beg='blue' -v end='gray' 'NR==FNR{new = new $0 ORS; next} $0~end{f=0} !f{print} $0~beg{printf "%s",new; f=1}' new.txt original.txt
ad then you can change them:
$ awk -v beg='water' -v end='tree' 'NR==FNR{new = new $0 ORS; next} $0~end{f=0} !f{print} $0~beg{printf "%s",new; f=1}' new.txt original.txt
red
blue
water
green
black
yellow
purple
tree
gray
white
Anything you might want to do is just a simple rearrangement using the same constructs, e.g. to not print the beginning line:
$ awk -v beg='blue' -v end='gray' 'NR==FNR{new = new $0 ORS; next} $0~end{f=0} $0~beg{printf "%s",new; f=1} !f{print}' new.txt original.txt
red
green
black
yellow
purple
gray
white
and similar tweaks to not print the ending line or not print both or do anything else....