Finding files which contain a certain string using find (1) and grep (1)

83

42

find . -type f -print -exec cat {} \; | grep some string

Command above doesn't print the file path.

I'm using: Ubuntu, bash 4.

H.Rabiee

Posted 2013-07-02T09:24:45.720

Reputation: 933

You may simply want to use ack to grep through all files under the current directory, possibly filtering on certain file types only. – Jaap Eldering – 2015-10-28T20:01:02.563

Maybe it does not meet your expected results because of the grep filter. What about this: find . -type f -fprint /dev/stderr -exec cat {} \; | grep some string – artistoex – 2013-07-02T09:44:24.603

What is the goal you want to achieve? On output you are getting grepped strings from files, you want path of those files? – mirkobrankovic – 2013-07-02T09:58:04.980

Yes I want to print the path of the files that contain the string 'some string'. @artistoex suggestion prints all file paths. – H.Rabiee – 2013-07-02T10:01:44.953

Answers

98

That's because you're feeding grep a stream of text which just happens to contain filenames. Since you provided no filenames as arguments to grep, it cannot be expected to deduce what file a matched line came from. Use xargs:

find . -type f -print | xargs grep "some string"

Since you have GNU find/xargs, this is a safer way for xargs to read filenames:

find . -type f -print0 | xargs -0 grep "some string"

If you only want the filenames that have a matching line without showing the matching line:

find . -type f -print0 | xargs -0 grep -l "some string"

glenn jackman

Posted 2013-07-02T09:24:45.720

Reputation: 18 546

How would I combine the above commands with search & replace? As in, grep then sed? – CMCDragonkai – 2015-05-21T14:27:36.157

1the find|grep pipeline outputs filenames, so: sed '...' $(find ... | xargs grep -l ...) – glenn jackman – 2015-05-21T15:31:56.757

117

I use

grep "some string" . -R

and it working faster

p.s.

More complex use case

grep -HiRE "some string|other string" . #H for file printing, i for case-insensitive, R for recursive search, E for regex 

To read param i explanation

grep --help | grep -- -i

ArcherGodson

Posted 2013-07-02T09:24:45.720

Reputation: 1 387

1This should be marked as the correct answer, to be honest. The currently selected one details something else entirely. – None – 2014-12-31T11:21:51.547

I'm so glad I kept reading. thanks! – catbadger – 2019-11-21T15:31:37.217

18And -n if you want line numbers. +1 for simplified answer. – Chad Skeeters – 2013-07-03T16:49:45.007

@ChadSkeeters exactly. Using the find does not give such solution. – ArcherGodson – 2013-07-09T12:45:00.273

14

I often search for source code in complex folder structures and I find useful using:

cd /your/folder/
grep -rHino "your string"

With those parameters, without using find, I obtain the file full path and the line number that contains the specified string.

It is also easy to remember because it BASHes through your search like a rHino :)

I will show how this works with a quick example.

Let's display the content of a file using cat:

jeeves ~ # cat fw.stop
#!/bin/sh
echo "Stopping firewall and allowing everyone..."
iptables -F
iptables -X
iptables -t nat -F
iptables -t nat -X
iptables -t mangle -F
iptables -t mangle -X
iptables -P INPUT ACCEPT
iptables -P FORWARD ACCEPT
iptables -P OUTPUT ACCEPT

And let's search recursively for all the files containing the string "iptables -P":

jeeves ~ # grep -rinHo "iptables -P"
fw.stop:9:iptables -P
fw.stop:10:iptables -P
fw.stop:11:iptables -P

As you can see in the output we have filename:hit row:searched string

Here's a more detailed description of the parameters used:

-r For each directory operand, read and process all files in that directory, recursively. Follow symbolic links on the command line, but skip symlinks that are encountered recursively. Note that if no file operand is given, grep searches the working directory. This is the same as the ‘--directories=recurse’ option.

-i Print the file name for each match. This is the default when there is more than one file to search.

-n Prefix each line of output with the 1-based line number within its input file. (-n is specified by POSIX.)

-H Print the file name for each match. This is the default when there is more than one file to search.

-o Print only the matched (non-empty) parts of matching lines, with each such part on a separate output line. Output lines use the same delimiters as input, and delimiters are null bytes if -z (--null-data) is also used (see Other Options).

Pitto

Posted 2013-07-02T09:24:45.720

Reputation: 1 766

2

I prefered

| find the file |make grep ont finding file | show de full path of the file

find / -type f -exec grep 'some string' {} \; -exec echo {} \;

user515185

Posted 2013-07-02T09:24:45.720

Reputation: 21

0

Silver Searcher is a very fast and handy util to search for files and content.

To solve your problem the silver searcher command would look like this...

ag 'some string' -l

-l Only print filenames that contain matches (don't print the matching lines)

qoomon

Posted 2013-07-02T09:24:45.720

Reputation: 181