Though sed has facilities to manage multi-line operations, they can be hard to understand initially. Perl can offer an easier set of tools.
A few weeks ago, working on migrating static versions of Pathology Lab and Histology Lab to Hugo sites, I had the task of swapping the position of an image and a subsequent div across 100+ Markdown files. Initially I thought that sed would be the right tool for the job of an in-place edit, but I found it confusing to manage what essentially is a multi-line regex.
To give some context, I generated the Markdown files using the very capable Scrapy framework. I used Scrapy’s stock html parser to fill in my front matter and simply dumped html markup into the body of each Markdown file. I knew that I would want to leverage some of the existing class names for layout, so I didn’t want to convert everything to straight Markdown.
So given all these Markdown files, though filled with HTML, my first thought was to use sed’s in-place edit capabilities:
sed -i.bak -E 's/match/replace/g'
The -i
flag makes sed create a backup file with the extension given, in this
case .bak
. The extension is required on MacOS. The -E
flag tells sed to use
extended regular expressions rather than basic RE’s.
This works great for a few words but on its own doesn’t work with multiple lines. The default behavior of sed is to match up to the new line. To approximate a multi-line match you have to copy input from the pattern space to a hold space, and then read it back out. It’s a lot to deal with.
So where does Perl fit into this? First, there’s little mystery about which flavor of regex Perl uses, and there’s plenty of tools out there to work out the regex first, Regex101 being my favorite.
But wait, you ask, isn’t Perl a language, not some unix command? Perl has a few
ways of reading in programs, one being
line-by-line on the command line using
-e
. Adding -p
makes Perl behave like sed, looping over the input fed to it,
and printing out the result. But we don’t want Perl to behave exactly like sed,
or else we’re back to matching line-by-line. Perlrun has a way to set a switch
to modify how Perl “slurps in”
files. In this case, the convention is to -0777
to slurp in files whole.
Putting this all together, connected to find
with -exec
, looks like this:
find . -name "*.md" -exec perl -0777 -i.bak -pe 's/<regex>/<replacement>/<flags>' {} +
If all goes well, you’ll have matching backup files next to the modified files.
Given that backup files are named *.bak
, you can use find and xargs to
cleanup (or find -exec if so inclined): find . *.bak | xargs rm
One important thing to note: though regex’s are quite useful, if you’re familiar with the Chomsky hierarchy of languages, you know that their power is limited. Particularly, regexes can’t parse HTML, as evidenced by this well known Stack Overflow post. If your match pattern extends over a variable number of nested tags, then it might be worth looking at other tools to parse the markup before moving it around. If Python is your thing, Beautiful Soup is a great place to start.
Printed from: https://compiled.ctl.columbia.edu/articles/better-sed-with-perl/