I've used Spacemacs for some years now as my primary editor, but recently I've been looking into #neovim.

It starts up a lot quicker, even with a bunch of plugins, and surprisingly for a terminal editor it has a more sophisticated UI than windowed Emacs.

@weavejester I’ve been using #vim for nearly 30 years :facepalm:. #neovim has re-wired my workflow.
Yes, #emacs fans will tell you the same about their #editor — but these tools aren’t just rivals, they represent opposing philosophies. Vim/Neovim: speed, composability, the Unix ethos. Emacs: the “OS inside your editor,” everything bundled in. Both have strengths, but they reveal two very different ways of thinking about code, control, and workflow.

@demiguru

I understand that emacs is designed to be invoked and then one will live in it until the end of his session. While vi is designed to be invoked as needed.

But on today's computer, emacs usually is also super fast, whether their invocation, opening files or when processing something.

I understand that emacs is not following Unix philosophy at all.

But in a sense, emacs is nothing but elisp interpreter with some addition to aid text editing.

All of those elisp packages are not emacs. They are just elisp packages, running on emacs. And those elisp packages usually are highly specialized tools.

I understand composability means each tools can be combined to build a more complex solution, such as through piping and redirection.

In a sense, elisp packages are just functions. We can use them in our own elisp code. But I know, it is different with the Unix composability.

#GNUEmacs #Emacs

@weavejester

@restorante @demiguru @weavejester

Vi isn't composable either. Sed and Ed and awk are composable.

That argument is out of context when talking about things with user interfaces.

I would say that emacs is ultimately more composable than vi, vim, or neovim. Emacs can integrate easily with other systems and tools.

That's the thing that it does amazingly well. Just because it's a super power doesn't mean that it's not within the philosophy.

The extensibility of #Emacs through integration with other tools is off the charts. That to me is completely within the Unix philosophy. I've been a Unix dev for 45 years. I think I know.

@Zenie @restorante @weavejester The claim that #Emacs is “more composable” than vi/#vim/neovim rests on stretching the meaning of composability. What Unix traditionally meant by composability was the ability to take simple, single-purpose tools and connect them together via well-defined interfaces (stdin/stdout, pipes, files). By that definition, ed, sed, and awk are indeed composable—they were designed to slot into pipelines without friction.

@Zenie @restorante @weavejester Vi (and later vim/neovim) may not fit that classical mold perfectly, but they do adhere to it in a limited way:

They operate directly on plain text files (not opaque binary state).
They can be scripted through ex commands.
They can be invoked non-interactively to perform transformations (vi -es).

That’s composability in the Unix sense: predictable input/output behavior and scriptable interfaces.

@Zenie @restorante @weavejester Emacs, on the other hand, is extensible, but extensibility is not the same as composability. Extensibility often means pulling other tools into Emacs, wrapping them in Lisp, and effectively making Emacs the hub of everything.
@Zenie @restorante @weavejester That’s closer to integration or even absorption, not composition. You don’t pipe the output of grep into Emacs and get a text transformation out the other side—you embed grep within Emacs as a Lisp function call.

@Zenie @restorante @weavejester That’s philosophically different.

Vim is composable in the Unix tradition because it’s file-centric, scriptable, and plays in pipelines.
Emacs is extensible and integrative, but its model of absorbing everything into itself arguably violates the spirit of “do one thing well.”

In other words, if we’re being strict about Unix philosophy, Emacs is powerful, but it’s less “composable” than vim—it’s more of an operating system in itself.

@demiguru

I am not an expert of elisp. But may be you are interested to:

https://www.gnu.org/software/emacs/manual/html_node/emacs/Command-Example.html

echo "Hello World" | emacs --batch --eval '(progn (insert (read-from-minibuffer "Input: ")) (write-region (point-min) (point-max) "output.txt"))'

I just simply copy paste the code above

@Zenie @weavejester

Command Example (GNU Emacs Manual)

Command Example (GNU Emacs Manual)

@restorante @Zenie @weavejester

What you’ve shown is Emacs scripting, which is fine, but it’s internal extensibility — you’re writing Emacs Lisp that runs inside Emacs. That’s not the same as taking existing Unix tools and composing them through standard input/output.

In your example: echo "Hello World" | emacs --batch --eval '…'

@restorante @Zenie @weavejester

Notice that echo isn’t actually feeding data through stdin to Emacs in a composable pipeline — instead Emacs is prompting and inserting via Elisp. Compare that to: echo "Hello World" | sed 's/World/Unix/'

@restorante @Zenie @weavejester echo "Hello World" | awk '{ print toupper($0) }'

Here, the tools are directly transforming streams, no custom scripting language needed.

@demiguru @restorante @Zenie @weavejester awk or sed is your scripting language in that case (much simpler and more limited than emacs lisp but still). 's/World/Unix' is your script, and you could write that to a file, use more complex or multiple commands etc. With emacs you can achieve the same thing and make it more composable by making a few adjustments to the example above:

@demiguru @restorante @Zenie @weavejester

echo "Hello World" | emacs --batch --eval '(progn (insert (read-from-minibuffer "")) (princ (string-replace "World" "Unix" (buffer-string))))'

This does the same as 's/World/Unix/' in sed and prints it again to standard output. It's not that #emacs is not composable like that, but that it's usually used interactively (because that's more fun).

@demiguru @restorante @Zenie @weavejester also, I think of #emacs more like a universal interface to your computer, the command line, all the tools you call etc. And in that sense I would interpret the unix philosophy a bit differently: the individual tools, whether that's a command line program or a lisp function, should do one thing well and be composable, but IMO it doesn't matter if you pipe them together on the command line or you combine them within emacs.

@eruwero
I get that view — #Emacs can feel like a universal interface to the system, and its #Lisp functions are composable in their own right.

Where I see the #Unix philosophy diverge is in the boundary: tools designed as separate executables, with contracts enforced by the #OS, versus composition inside a single runtime. Both routes achieve interoperability, but one leans on external guarantees, the other on internal extensibility.

@restorante @Zenie @weavejester

@demiguru @restorante @Zenie @weavejester IMO this internal/external distinction is relatively arbitrary. In one case you have "separate executables" that you call from a shell, which is equivalent to calling a function, with the shell being the interpreter. In the other case you have lisp functions and a lisp interpreter (emacs) to call them.
@eruwero @restorante @Zenie @weavejester Right, that’s the analogy I was pointing at — a #shell is an interpreter, processes are its “functions.”
The key difference is that in the #Unix model, the #OS enforces the separation: each process is isolated, communicates through defined streams, and can be swapped out independently. In #Emacs, #Lisp functions share one runtime. Both are composable, but the guarantees differ.
@demiguru @eruwero @restorante @Zenie @weavejester for me this sounds not GNU Emacs specific. Most/many Lisp implementations on UNIX will work that way: code & data are shared in a single address space and provide a resident programming system.
@demiguru @eruwero @restorante @Zenie @weavejester the typical Lisp Machine then had only one Lisp runtime with one address space, booted as the operating system
@symbolics @eruwero @restorante @Zenie @weavejester Right — and that’s why I see #Emacs as #Lisp-lineage first, #Unix-lineage second.
@Zach  🇮🇱 🇺🇸

I think you're basically correct to point out that Emacs does not adhere to the unix philosophy as it is classically understood.

The locus classicus is presumably the widely quoted McIlroy et. al. in the 'Unix Time-Sharing System" where Emacs clearly violates the first maxim:

Make each program do one thing well. To do a new job, build afresh rather than complicate old programs by adding new "features."

Or to take a later articulation, Gancarz in Linux and the Unix Philosophy's first two tenets: 'Small is Beautiful', 'Each Program Does One Thing Well'  — again Emacs is the opposite of this.

(This kind of thing comes up constantly in every mention of the Unix philosophy that I've ever seen until this conversation and is easily verified by simply searching.  It's strange to see people insisting that there is simply no emphasis on small, special purpose programs. Of course one can decide that principle isn't important, but let's not ignore that it was (and I believe usually still is) considered to be so)

We can also note that in the original incarnation of Unix ('Research Unix') there was nothing remotely like emacs and there was, as far as I know, very little scriptability.  The Thompson shell allowed for piping but not scripting.  OG unix is everything is written in C, with a simple interactive shell and if you want anything else it's back to C programming.

Whereas Emacs does (as you say) have more in common with the Lisp ethos, where there aren't separate programs per se and everything is a function in one giant Lisp environment (same is true of Smalltalk) — although there are differences there, too which I might get on to later.

I think it pays to note that these views are informed by very different language traditions.  C is a static language with an explicit and slow compilation step.  Lisp on the other hand has always been a dynamic language where you can just throw a new function into your environment pretty much immediately, usually without having to (explicitly) compile or restart anything.

If you think about it a bit you might start to see that if all you've got is C, small programs that don't keep their own state but write to text files may be the simplest option for getting any composibility, particularly on limited machines.

However, I think we can fairly say that Unix found that only having a statically compiled language available was far too limiting and early on started haltingly down a path to acquiring dynamic programming abilities, starting with scriptable shells, and later larger programs with their own scripting languages.

People often cite the Unix philosophy as though it's obviously the right way to program.  Modularity, clear interfaces, and composibilty we all think are good things, of course.  But why is 'one program does one thing' better - or even equal to - 'one function does one thing'?

#emacs #lisp #unix
Zotum

@jamie

Appreciate the way you framed this — especially grounding it in McIlroy and Gancarz. That’s exactly the lineage I was pointing at: #Emacs diverges from the “small tools, one job” maxim because it inherits from the #Lisp/#Smalltalk ethos instead.
#unix

@jamie I agree it’s less about which is better and more about recognizing two different language traditions: #C pushing toward external composability through processes, #Lisp pushing toward dynamic, internal extensibility. Both solve real problems — just with different assumptions about boundaries.

#unix

@Zach  🇮🇱 🇺🇸

To continue from where I left off, the pure Unix conception didn't envisage dynamic languages and scriptability and explicitly eshewed large programs.  The utility of (some) dynamism was realised early, but corrected in hamstrung sort of ways — especially from a Lisp perspective,  but anyone can see the limitations of shell scripting languages.

And the 'small programs' in practice has not played out either.  X isn't small, vim isn't small (not by trad Unix standards - it's similar order of magnitude to emacs in fact), webbrowser aren't small...  

The other side of the equation is that the Lisp (esp. in the form of Lisp Machines)/Smalltalk world supposed that you would live entirely within the environment and everything would be written in the one language.  The notion of interoperating with anything outside this was alien and resisted.  It was some years, I believe, before a C compiler became available for Lisp Machines.

Emacs transcends all of this.  It participates in the Unix world just fine.  If you want to load, edit something, and exit, you can do that (if it's running in the background as a server, this is plenty fast, too, as you're just executing emacsclient).  If you want to use it as a filter on the command line, it can do that too, as has been demonstrated.

And the unix world is also available from within emacs.  You can filter or send any block of text through external commands, it has wrappers for classic unix tools like grep, also modes for interacting with shells, terminal emulators, and its own shell from which you can run both elisp functions and unix commands together.

(And, of course, it's great at handling Unix's common data format: text.)

Moreover, org-mode has a great multilingual environment where you run source code in numerous languages and even pass data between them with ease.  


This is all completely the opposite of the Lisp Machine/Smalltalk notion of an isolated world.  It's more in line with the Unix shell, which doesn't care what your programs are written, except better.

So I think we can say it transcends the two purisms.  It does this by rejecting the 'small specialized program' dictum (which I don't think is important and I'd argue has been effectively rejected by all but the most purist Unix-heads), and centering itself in place of a unix shell (without demanding this) but keeping to the spirit of all the other ideas.   And on the other hand, also rejecting Lisp Machine solipsism.

(I haven't really thought of this in quite this way before so thanks for the discussion, in particular @Zenie who pushed the idea that emacs is actually better at what's important about unix than unix is, and @restorante and @eruwero's demonstration of emacs as a unix command line tool)

#emacs #lisp #unix
Zotum

@jamie

That’s a solid synthesis — I like how you framed #Emacs as transcending both the #Lisp-machine isolation and the #Unix small-tools minimalism. Glad the exchange led somewhere new.

@Zach  🇮🇱 🇺🇸

I did want to come back to my points that few actually follow the pure unix way any more, and that Vim isn't a small program by Unix standards.

As an indication(*), I've started terminal vim, neovim, and emacs -nw -Q processes, and KDE's system monitor reports the following:

nvim      3.0 MiB
vim      11.9 MiB
emacs 40.3 MiB

All of these are large or worse compared with original unix tools.  I think McIlroy of 1978 would be horrified by all of them. Classic vi was 160kB or so later in its life.   Bill Joy described ex as 'very large' in 1979.  

They also all have scripting abilities and many packages of extensions available (neovim less so it seems, but it's also newer)  None of them are really small, narrowly focused, or minimalistic by the standards of Unix in its first decade.

The classic Unix programming workflow as I understand it was to fire up an editor, edit, exit, recompile and retest.  

I think the numbers of people who do that today are small. I think people using the vi family also fire up their editor and leave it running, and often controlling things from the editor.  

(e.g. #^https://www.reddit.com/r/vim/comments/j33ow1/what_is_your_vim_workflow_like/  )

Also, once you've got a decent scripting language, you've already taken one step on the emacs path.  

So compared with classic Unix ideas, they're all alike.  They're all large, they're all scriptable, they all have libraries of plugins, and they all enable if not promote a form of use where you live in the editor, not the shell.  The only real difference is that the vi-family communities haven't gone as far as writing extensions —but that's cultural, the software itself allows for this.

#unix #vim #vi #emacs
--
(*) what would be a fair comparison?  There seem to be many possibilities. If one is trying to get the smallest possible memory footprint and is prepared to work for it, of course the size of emacs could be reduced, perhaps substantially.  

But if we're less concerned with individual binaries and more 'how much do we need to get equivalent functionality' then we'd have to look at vim + mutt + tmux + fish + lynx + ...

@jamie

You’re right — pure #Unix minimalism couldn’t survive once systems became interactive and persistent. Early Unix assumed short-lived processes and text as shared state.
Once #editors became programmable and long-running, that boundary blurred. #Vi/#Neovim kept the file-centric DNA; #Emacs internalized it. Both descend from Unix, just different branches of its evolution.

@jamie

Thank you saying this so well.

@Zenie

I've been meaning to ask you:  why do you consider vim to be a poor vi?  And why is Evil (if I've remembered the right mode) better?

(I don't have any opinions about this; I'm just interested)

#vi
Zotum

@jamie

I think it just my experience. I was a vi user many years before the vimulator came along. I didn't like the key-bindings for buffers and the developer wasn't nice. I stuck with my ways in vi until 95.
I switched to emacs but used viper the vi emulator in emacs. Emacs buffers and keymaps are amazing compared to vim. Eventually I kept a reasonably configured vim around. But I'd never use it for more than a quick look at something. 2011 I used vim for a year, for work. It was very lacking despite a ton of vim script. Which is awful. I switched back to emacs with evil.
Recently I've realized I don't like evil is it's goal to emulate vim.

@jamie
So really it comes down to my view that #Vim is emulating #Vi, and so is #Emacs.

If you then look at the infrastructure around that,
Emacs is much more capable and less limiting in what you can do. That everything is a buffer means that it's buffer management is amazing.
The modes and key-bindings are awesome and it's all
pleasantly programmable in lisp.

Vi is a great editor, but there isn't much to it. It's a lightweight editor that works well. I think people confuse what is emulating Vi versus what is the application surrounding the emulator.

Emacs has, Evil, Viper, Meow, Boon and Meep modal editors. That says something too.

@Zenie Have you taken a look at Neovim? I've recently been trying it out after a decade of Emacs, and I've been surprised by how capable it is. I don't think there's much between modern Emacs and modern Neovim in terms of capability... unless you're using org-mode.
@weavejester
I use neovim as my vi. I adopted it in the beginning.
I use it as it comes out of the box.
At the time it wasn't really very scriptable yet and I haven't looked again. Maybe if I programmed it in ferret. 🙂
@Zenie

I see.  So it's not that vim is a bad emulator of vi, it's just that outside of emulating vi it doesn't have much to show for itself.  Gotcha.

#vi
Zotum

@Zenie

On the 'confusion' note, these days I'm increasingly of the opinion that a lot of rationales for why someone prefers a particular piece of software (and a lot of other things) aren't genuine reasons at all but just confabulations for what is really being done for aesthetic or even cultural reasons.

E.g. one likes the idea of small, and vi was small, so you use vim to symbolically associate yourself with smallness, rather than interrogating why (or even whether) small is good, how small is vim really, what counts as small on modern machines, what are the solid practical advantages of your preferred tools over stuff that makes you feel good, the whys of which you can't really articulate...
@jamie
I see people arguing for their limitations everywhere.
A lot of it comes from their point of view. They can't see what's in front of them or what would benefit them because they need to step back and to the side.