Arcuru 12 hours ago

Using `cat file.txt | ...` has always felt more natural to me, so I still use it.

If I need to build a long command I've been using the excellent `up` tool to do it, e.g. `cat file.txt | up`

https://github.com/akavel/up

  • REDS1736 11 hours ago

    Wow that looks amazing, thanks for telling me about this!

    • agumonkey 2 hours ago

      Indeed. A good addition to fzf and bat

codesnik 11 hours ago

if you prefer to use cat just to read your pipeline from left to right, you can just start your command with file redirection. It doesn't have to be at the end of the command.

  <file grep something | do something else ...
this works in bash too, but if you're using zsh, there're a couple of nice shortcuts

  <file
on it's own works as

  more file
and

  > file
  ...
  ...
  ^D
allows you to put something into the file quickly without firing up the editor. Though

  > file << end
  ...
  end
will give you nicer line editing experience.

more zsh redirection tricks here: https://zsh.sourceforge.io/Doc/Release/Redirection.html#Mult...

  • gorgoiler 2 hours ago

    It’s shell specific though, just like using FOO=bar cmd prefixing to make environment changes. Better to use env(1) and keep everything rooted in Unix processes rather than shell syntax, imho.

      # bashisms
      <file tr z-o m-g | …
      FOO=bar python3 …
    
      # processes
      cat file | tr z-o m-g | …
      env FOO=bar python3 …
    • MathMonkeyMan 6 minutes ago

      Both of the features that you mention are specified in POSIX, which means that bash, ksh, zsh, sh, ash, dash, etc. implement them.

      I find myself googling "OpenGroup shell command language" pretty often to check this sort of thing.

  • chgs 11 hours ago

    Or you could just use cat

    • mingus88 10 hours ago

      Coward, use `tail -c +0` like the rest of us useless command cowboys

  • globular-toast 11 hours ago

    If I saw

        <file grep ...
    
    It would give me pause. If I saw

        cat file | grep ...
    
    I would understand it instantly, as would any other Unix user. Therefore the latter is better code.
    • zahlman 10 hours ago

      `<file` at the start might not be idiomatic, but when I see it I think it makes perfect sense. It puts the pipeline "in order", starting with redirection from the input (and presumably ending with redirection to the output, if not stdout).

      • HappMacDonald an hour ago

        I usually interpret `command <fileA >fileB` as meaning "shove fileA into this command, and shove output into fileB". The "arrows" make visual sense this way.

        `<fileA command` OTOH at least visually gives off the impression that it's sending the command's output to the left (changing the prompt, perhaps?)

    • DevilStuff 2 hours ago

      Better code, yeah I agree. It's way more easily understandable. At least for me though, 99% of my bash use is stuff that will never be seen by anyone else, so the simpler keystrokes one may prove useful. Maybe if I get used to it :)

    • CBLT 11 hours ago

      I don't feel <file is obscure. I've seen shell in that style from coworkers and from open source. Your value judgement against it might just be your experience, rather than something universal.

      • strunz 8 hours ago

        It's definitely more obscure than cat, and even those familiar might be confused with it at the beginning of the line

      • g-b-r 9 hours ago

        For sure many people don't know what it does

        Furthermore, a little memory lapse or typo, and, oops! >byebyefile

    • sgarland 10 hours ago

      Or just use

          grep <pattern> file?
      • pastage 2 hours ago

        The point is you do not know if file is a stream, an archive, too big, currently empty or just contains bad data for your current test. So using cat file is a way to standardize debugging or make development easier.

    • AStonesThrow 2 hours ago

      Both of those invocations would give me pause, because grep is a command that can handle file arguments on its own; in fact it can incorporate knowledge of those filenames into its output quite usefully, so I would generally avoid redirects into grep when I could use it to explicitly, directly, open the files in question.

    • crdrost 7 hours ago

      The problem is that according to this argument `cat` should never be used for the thing it was designed for, because if you use it with more than one argument, it would give most Unix users pause, but if you saw it with only one argument they would understand it instantly.

      The whole point of the "never use cat" Unix advice is a war between instrumentalism and design purism. Should things be known mostly for what they're useful for, or mostly for what they were made to do? If you understand this, then the war is not soluble but you can at least phrase a third position that will reconcile both sides. If you don't understand this, then the war is over an issue that doesn't even exist.

      Hot take: `<file> | abc` and `abc | <file>` should both make sense because a file should be understood by default as a command that reads from and writes to a particular destination, and the shell should take it seriously that `abc | <file>` needs to be easily undoable if the file wasn't chmodded +x appropriately.

      • oktoberpaard an hour ago

        So `abc | <file>` would write the stdout of `abc` to the file? Meaning that if you forgot to mark your script as executable, you’ve now lost all your work? I think `abc > <file>` works perfectly fine and pipes don’t need to substitute this behavior.

        • crdrost 22 minutes ago

          That's the point is that it's up to the shell to make sure that you don't use your work if you do the wrong thing. Could be as simple as prompting you to make sure -- "Overwrite <filename>? [yn]: " -- or could be something where the old inode is still available under some magic path for a few more commands, or could be that you use special syntax for the command, like "@<path>" is the stdin/stdout command form for the path... or it could be something else entirely, really.

          For an instance where there are thousands of bash scripts out there that work around a case where `abc >file` does not work, consider the times when you want to write to a file which you don't have write permissions to, with sudo. my favorite way to do this is with `tee` but I have also seen others: but `>file` is not one of them! Because it's emphatically not part of the algebra of the rest of the shell; the rest of the shell is written in pipes, this one command says "I'm going to make your pipeline terminate, this step will be un-pipeable."

          For another example where the algebra doesn't consistently handle the whole system, consider that in a normal language you would consider writing a function which returns a value and also prints debug logs to the process's stdout. Bash can't do this on its own, you have to figure out which /dev/tty is attached to the process and then write your debug log to that TTY, and then your script probably fails in interesting ways when it itself is redirected to a file and there is no TTY that it is attached to anymore...

      • pastage 2 hours ago

        Can you explain what fou mean by <file> | abc? I have no clue what that would mean. Further what does undoable mean in a shell context?

      • makapuf 2 hours ago

        What if you want to grep a bash script file?

kelnos 11 hours ago

People who get up in arms about "useless" uses of cat need to chill out. I read my pipelines left to right. Using a '<' redirection to pull the data in after the command means I have to swap my reading direction back and forth, and why bother with that? I have better things to do with my brain energy.

  • bloppe 11 hours ago

    As others have mentioned, you can put `<` anywhere in the command, including at the very beginning. But, I agree that getting up in arms about this is really dumb.

sim7c00 12 hours ago

cat all the things. i cat | grep and im not ashamed of it! cat | less - why not.? its not going to melt my pc more or less if i cat | more | less...

echoangle 12 hours ago

> Because for me it'd only be a local maxima.

Nitpick: this should be „maximum“, maxima is the plural.

  • smitty1e 10 hours ago

    Note the 'local'. If the output has several humps, multiple minima/maxima are possible.

    • zahlman 10 hours ago

      Yes, but "it" and "a" in the original sentence are singular.

terminalbraid 19 hours ago

My main use of cat is for clipboard pasting (if I can't leverage pbpaste or something more appropriate). If I want to get something into a file or get into a pipe as stdin `cat > file` or `cat | cmd`, paste, then end with cmd-D to send an EOF and terminate the cat. For files it circumvents "helpful" formatting you would get pasting into an editor.

  • codesnik 11 hours ago

    if you're on os x, you can also do

      pbpaste > file
    
    and if you're using zsh, just

      > file
    
    is sufficient, without cat
js2 11 hours ago

> Some people insist that we should never use a cat

Rather, people argue against useless use of cat, which is different. Personally, I'd never argue against it on the command-line even though some do. It's once you commit your pipeline to a script where you may as well remove it.

There are also suitable uses of cat that I don't think anyone would argue against. Besides the obvious use of outputting one or more files, it's also often used with here docs.

its-summertime 11 hours ago

The underlying topic for this is that, most shells treat files/redirections as part of the individual commands and not part of the pipelines. Files should be sometimes treated as commands instead, they plug into the fds of a process the same way as commands do.

`cat x | head` attaches `cat x`'s stdout to `head`'s stdin, `< x head` just attaches `x`'s fd to stdin. In this way, treating a file like a "light-weight" command (light-weight because no backing process) makes more sense, `open x | head` perhaps

Though I will note that, in a way, bash supports this, as it has a modules system, and one of the pre-existing modules in the source tree is a `cat` replacement, through my lack of understanding of C prevents me from working out if its optimal

https://git.savannah.gnu.org/cgit/bash.git/tree/examples/loa...

  • rolfan 10 hours ago

    From the file you linked, the built-in just copies the input files one by one, sequentially, into stdin. All done in chunks of 4096 bytes, a common page size.

    Is it as optimal as just attaching the file descriptor? No, because you will have to pipe the contents. Is it optimal (or at least optimal enough)? Considering how pipes work, yes.

fifilura 13 hours ago

"A la test driven development. "

I would not call that test driven development (where are the testcases?)

Rather more like REPL driven programming.

  • delichon 12 hours ago

    Whether there are formal examples or not, the thing you're doing during REPL driven programming is testing. It's just rapid, iterative and informal.

    • bityard 12 hours ago

      TDD refers to a specific kind of testing methodology, not just any and all testing. Manual testing is not TDD.

userbinator 11 hours ago

The logic of the author's argument seems rather convoluted. To me, cat or not is a simple decision: does the first useful command in the pipeline accept input from a file, or only stdin?

ttyprintk 19 hours ago

A better example is ‘pv -l’. Seeing cat in a heavily-used pipeline means pv is not available.

wwalexander 12 hours ago

You can still just begin with <somefile instead of cat somefile |. Does it really matter? No, but I don’t think pointing out useful and concise built-ins is necessarily asinine.

  • jchw 12 hours ago

    Yeah, there are still some edge cases where cat is useful, but more often than not I just begin the redirection. I reckon some don't realize you can put it at the beginning, which does sometimes feel more ergonomic in a pipeline.

    • JoshTriplett 12 hours ago

      It's not that it feels awkward to write the redirection first. It's that it feels awkward to write `< file cmd`, with no operator between the filename and the command, and the redirection operator not pointing at the command.

    • kortilla 12 hours ago

      A pipeline without pipes connecting is just a field of shit

  • kelnos 11 hours ago

    Can you? In bash I end up with no input at all if I start the line with '<somefile'.

  • AStonesThrow 12 hours ago

    No, in ordinary usage it doesn't matter, and it's not like I would go out of my way to change someone else's script code.

    But if teaching or demonstrating shell features, I'd say that it's important to know that redirection is the most efficient, explicit, and available method.

    Indeed, a student could learn "cat file | ..." as a standard idiom and never use an initial redirect, but what happens when they come across one "in the wild"? They should know its proper interpretation, because it can otherwise be a bit jarring to see and difficult to interpret.

PLenz 12 hours ago

I honestly use tac more then cat

  • yjftsjthsd-h 12 hours ago

    I'm not sure I have ever used tac for real. Could you give an example?

    • chgs 11 hours ago

      Reading a log file from most recent entry for example

      Cat *log|grep error|tac for example

      Tail -500 log | tac

    • sgarland 10 hours ago

      When you want to tail a file, but you want it in descending order and you’re not sure how much of it you want:

          tac file | less
      • yjftsjthsd-h 10 hours ago

        I guess the difference is that I expect things to read top to bottom, so I'd do that as

          less foo.log
          G
        • sgarland 10 hours ago

          Fair enough, if that’s how you want it to read. less on its own is certainly quite capable.

  • shaftoe444 12 hours ago

    In my experience it's best to do

      cat file | tac
tptacek 12 hours ago

Does anyone take the "don't use cat" stuff seriously?

  • prmoustache 9 minutes ago

    Depends what you mean by seriously.

    If by that you mean religiously avoiding cat at all cost, no. If by that you mean thinking about what you are actually doing on your commandline I say I do. Which makes me realize that yes, cat is more often than not redundant unless you are actually outputting the concatenation of multiple files.

  • gorgoiler 2 hours ago

    It’s a hump curve with virtue signalling in the middle, like a thousand other things in life.

    Noob+naive: use cat because don’t know better.

    Experienced+smart ass: use shell direction to distinguish oneself from noobs.

    Older+wiser: use cat again because it flows left to right, isn’t bash specific like “<file foo”, doesnt special case examples with only one file as input, and allows something other than a file as input or lifting the body of the pipeline into a function without changing the text…

    …but mostly because it doesn’t make one come across as a nit picking asshole to the expensive noobs you’ve just spent months trying to hire.

  • Ferret7446 11 hours ago

    It's more of a symptom of someone who doesn't understand what's actually going on, i.e. cargo cult scripting.

    Specifically, it indicates some combination of:

    1. isn't aware of processes/pipes

    2. isn't aware of cat's "primary" functionality

    3. isn't aware of shell input redirection

    4. isn't aware that shell input redirection can be put before the command

    • cybrox 10 hours ago

      90% of the time, I care more about someone like that understanding my script than the tiny performance impact of an extra process or 'misusing' a function.

  • tux3 12 hours ago

    Well, this is the internet!

    We had just gotten over tabs vs spaces after barely a thousand years of noticing the other side is objectively wrong.

    Something new that you're definitely doing wrong to blog and argue and bicker about; it can't be helped!

globular-toast 11 hours ago

cat concatenates files together. I often use it for that. Who cares if "files" is just one file? Using cat regardless keeps things uniform. I like it. People who get funny about it are just trying to be clever. Yes I know there are other ways to do it. But you understood what cat did, otherwise you wouldn't have commented on it.

  • Svoka 11 hours ago

    literally what cat stands for. conCATenate.

nullc 9 hours ago

I usually start with cat -- later the head doesn't just get replace the cat may too, e.g. replacing it with some command whose output was only temporarily stored in a file for development purposes.

cocodill 11 hours ago

and a cat is also good against mice

smitty1e 10 hours ago

I'm a believer in the ditt-k-pow, DTTCPW, the Dumbest Thing That Could Possibly Work.

What, exactly, are the arguments against `cat`?

  • cybrox 10 hours ago

    The argument is that it's redundant because 'cat f | head' does the same as 'head f'

    I don't mind either way but I find starting with a cat and reading LTR to be very easy to understand.

x3n0ph3n3 11 hours ago

  cat | sudo tee somefile > /dev/null
is still my favorite way to paste text into an ssh session and save it to a protected file.
  • pbaam 9 hours ago

    The cat command can be omitted there, as tee reads from standard input by default, even if stdin points to a terminal. I was going to comment an actually useful (and unavoidable in bash) use of cat and ssh, which is to do do nothing with standard input and redirect it to a file:

      <file ssh 'cat >file'
    
    And you could just use scp, but I've found clients without scp and servers with the SFTP subsystem disabled.
    • x3n0ph3n3 6 hours ago

      I want do paste from my clipboard, not copy another file.