I won’t shut up about ed(1), don’t even ask. Because it’s a good text editor and IDE. And a perfect text computation playground. Bare regex acting on lines of text. What else could one want?
Oh, right. Turing completeness. That’s a hard one. It’s a text editor (I say with an indecypherable smile.) Not supposed to be Turing-complete, right?
Wrong. ed(1) is Turing complete. Because:
So meta. Infinite potential.
Now wait, we know that Turing-completeness requires branching.
So
In case you need something more… conventional than this weird regex mapping.
You can do early escapes/returns.
If you consider every ed(1) script a function of its own, it starts making sense.
Doing an action until the data is ready.
Two techniques there:
The perfect ed(1) script terminates when reaching the desired state of text.
Because that’s easy enough to encode.
And it plays well with recursion and self-calls.
Say what?
One task I had that seemingly didn’t fit ed(1) was file inclusion.
I use ed(1) to build my website.
Naturally, I need a way to include/expand templates.
Usually represented by separate files with variables in them.
I used an imaginary HTML tag
The logic is:
So the pattern is clear: process the anchor line/instruction into a set of commands.
Make these commands act on the arguments substituted from the instruction.
And call ed(1) on that set of commands.
Same technique is used in my shell execution instructions.
Basically tags of a form
Same technique, except it’s
Now that we learned to call ed(1) from inside ed(1)…
Why not call is some more?
Many times.
Maybe even infinite number of times!
Yes, I’m talking recursion.
Recursive scripts combine both conditionals and self-calls.
Conditionals are particularly important: they allow to bail out from recursion.
Instead of recurring indefinitely.
Here’s an example script that modifies itself until it has the necessary number of
I know, not the most practical script.
But hey, ed(1) was powerful enough for me to have no need for recursion!
Yet.
All the kudos goes to a cool anonymous person for this hack:
You can create N lines and act on them to repeat some modification on the buffer:
So yeah, ed(1) is a powerful enough programming system, supporting
I mean, even BASIC can’t do some of these.
And they used BASIC in eighties for Real Programming™.
They might’ve used ed(1) and end up better for that.
Don’t commit their mistakes.
Use ed(1).
P.S.
I know that ed(1) Turing-completeness was already proven.
The difference of the proof there and my post is that I’m actually using ed for practical things.
Most of the hacks here are the ones I made for my nefarious needs.
Enjoy.
P.P.S. Just for fun, here’s a super meta script:
P.P.P.S. There’s a whole treasure trove in
my ed(1) entries on RosettaCode,
but I leave reverse-engineering them to a more capable person.
Have a good time watching me dissolve in the madness!
Using
Self Calls
g/<include "*\\([^">]*[^"> -]\\{3\\}\\)"*\\/*>/s//&\\
H\\
\/<include\/d\\
-1r \\3\\
wq\\
/
g/<include "*\\([^">]*\\)"*\\/*>/d\\
.,+4w !ed %
E
Take lines with these
g/<exec "*\([^">]*[^"> -]\{3\}\)"*\/*>/s//&\\
H\\
\/<exec\/d\\
-1r !\\3\\
wq\\
/
g/<exec "*\([^">]*\)"*\/*>/d\\
.,+4w !ed %
E
Iter... Recursion!
H
$g/ɩ\\{70\\}/q
$s/$/ɩ
w
,w !cat % | ed -s %
Q
ɩ
a
1
2
3
data
data
data
.
1,3g/.*/4,$s/.*/&&/n
Q
!# Results in:
!# 6 datadata
!# 6 datadatadatadata
!# 6 datadatadatadatadatadatadatadata
ed Is (Turing) Complete
positive conditionals
!echo "HELLO" > hello.txt
E hello.txt
p
s/.*/p\\
s|.*|p\\\\\
s;.*;p\\\\\\\\\\\\\\
Q\\\\\\\\\\\\\\
;\\\\\
,w !ed %\\\\\
Q\\\\\
|\\
,w !ed %\\
Q\\
/
,w !ed %
Q