Counting lines with Go
I was thinking the other day about The Unix Philosophy. Broadly, you can get a lot of power from small command line utilities that do one thing well and can be chained, or composed, into more powerful use cases. For example, there's a command called wc
(word count). It, well, unsurprisingly, counts words. But, you can also have it count lines if you pass in a flag wc -l
.
I wanted to take the idea of "do one thing well" to the extreme. And, since I'm learning Go, I thought a line count utility would be a fitting exercise. In building this, we'll be able to learn how to read input from another program to support composability; we'll learn a bit about command line arguments; and we'll learn about reading files.
I mentioned composability a bit. Let's see what that entails. It means that this utility will output only the line count (so that it can be piped on to another command). And, it will need to accept, as input, either the output of another program (we might call this piped input). Or, it will need to be given a file location.
This means that our little application might be called like this
> lc "path/to/your/file.txt"
or like this
> echo "Count the lines in this" | lc
(you can substitute cat
, or grep
, or anything else, for echo
above)
In either case, the count of carriage returns (\n
) in the file will be printed out.
> lc "path/to/your/file.txt"
109
If you want to just see the code, it's available on GitHub.
Implementation Considerations
There are some counting assumptions that I made. I had originally chosen to have this match my editor's line count. That is, if Visual Studio Code shows x
lines then my logic would also show x
lines. However, I've chosen to follow the behavior of wc -l
. I count carriage returns (\n
). If a file does not end with a carriage return then the last line will not be counted.
While I'm not sure how I feel about this behavior, it is consistent with other tooling. A trailing carriage return is required to get an accurate count. Changing this is an exercise left to the reader.
Also, I think I might want to reuse the line-counting logic in other applications. So, I'm going to separate the command-line interface from the code that understands how to read through a stream and count returns.
Handling a file argument
Go has a nice flag library to read from the command line. We're going to abuse it a bit to read in the first argument a user passes. Remember, the first scenario is, lc "path/to/your/file.txt"
. In this usage, there are no named flags being passed in. I simply need arg0
.
flag.Parse()
filePath := flag.Arg(0)
if filePath == "" {
fmt.Println("Usage:\n\tlc \"path\\to\\file.txt\"")
return
}
file, err := os.OpenFile(filePath, os.O_RDONLY, 0444)
if err != nil {
log.Fatal(err)
}
defer file.Close()
countLines(file)
I'm opening a file for read-only access, and requiring that the file have read permissions set for the user, group, and other. We'll get a file descriptor back from the call to os.OpenFile
. You can think of a file descriptor as a small reference that we'll keep around so know know where to read data from later. And, to clean up after ourselves, we'll close the file when we're done reading. We can do with with the call to defer file.Close()
Handling piped input
The other use case we have to handle is when the data is passed, or piped, directly into our utility.
> echo "Count the lines in this" | lc
This is passed in on stdin
, or standard input
. Unix introduced three standard streams. These are ubiquitious enough that many programming languages have some way to access them and will, as needed, abstract the implementation details for the operating system away for you. These streams are stdin
, stdout
(standard output
), and stderr
(standard error
). You might read something from the user on stdin
, give them some results on stdout
, and log any problems to stderr
. But, as you can see in my use case above, stdin
might be input from anything, including another program.
And, Unix loves file descriptors. Like, it really loves them. stdin
, stdout
, and stderr
are all file descriptors. Yep, just like the result of the os.OpenFile
call earlier. This will come in handy. Trust me.
We need to probe the stdin
file descriptor to see if we received any data.
stat, err := os.Stdin.Stat()
if err != nil {
panic(err)
}
if stat.Mode() & os.ModeCharDevice == 0 {
reader := bufio.NewReader(os.Stdin)
countLines(reader)
}
Calling Stat()
will give us information on the associated file descriptor. In this case, that's Go's reference to stdin
, which Go nicely stores for us in a variable called os.Stdin
.
I know stat.Mode() & os.ModeCharDevice == 0
looks a little hairy. We're asking Go for the current file mode on the file information. This is a bitmask of the current modes that are set on the file. When the 'this is character input' flag is set, then we know that stdin
is open and being written to.
We could read directly from the stdin
file. But, I want to buffer the input to more efficiently traverse it. Go provides a buffered I/O library for just such a use case. We give bufio.NewReader
an io.Reader
and get back an io.Reader
. What a deal. But, it does a ton for us under the hood.
Now that we have an io.Reader
, we can call into the package (that we haven't written yet), and count them carriage returns.
func countLines(r io.Reader) {
count, err := lc.CountLines(r)
if err != nil {
log.Fatal(err)
}
fmt.Println(count)
}
Count Them Lines
Long article. I know. But, we're here. All of that setup and we can deliver the actual value in 22 lines of code
func CountLines(r io.Reader) (int, error) {
var count int
var read int
var err error
var target []byte = []byte("\n")
buffer := make([]byte, 32*1024)
for {
read, err = r.Read(buffer)
if err != nil {
break
}
count += bytes.Count(buffer[:read], target)
}
if err == io.EOF {
return count, nil
}
return count, err
}
Just to recap, we want to know how many times a \n
character appears in a given file. In a text file, at least, this would tell us how many lines long it is (emoji can't have carriage returns in the middle; I checked).
We don't care about any of the content. So, the question is. How can we efficiently read through the file, get what we want, and get out. I'm going to rule out using io.ReadBytes('\n')
or higher-level abstractions like scanner
as I want to hold as little in memory as possible. With those options, I might end up trying to read an entire file into memory before I find the first newline character.
But, we can create a byte buffer, read into that buffer, and then search just that buffer. When we're done, we'll move on to the next chunk.
So, let's create a 32 kibibyte buffer
buffer := make([]byte, 32*1024)
and then read from our file into that
read, err = r.Read(buffer)
This might give us a buffer with the following content
Hello World\nThis is a\nThree line file
The variable read
will tell us how many bytes were read in. This is very important information because we're reusing our buffer and not re-initializing it between reads. Suppose we read 32kb of data on the first call to r.Read(buffer)
but only 5kb of data on the second call. Our buffer will still contain 32kb of data. 5kb from the last read followed by 27kb of old data...
Next, we can use the bytes.Count
method in the Go standard library to find the number of newline characters. We'll store the result in our counter variable.
count += bytes.Count(buffer[:read], target)
We will loop "forever". In reality, we will read incrementially to the end of the file. Then, we'll try and read one more time and Go will return an end of file error. We'll check to see when this is encountered and return our results then. In our use case, an end of file error is expected so... well, we shouldn't return it to the caller.
if err == io.EOF {
return count, nil
}
return count, err
An Alternative Way to Read
Calling bytes.Count(buffer[:read], target)
is a very specific choice that I can make for this application. However, it might not always work for us. Suppose we were looking for a slightly more complicated pattern. Go has a way for us to do that. bytes.IndexByte
will return the index position of the first occurance of a byte in a byte array. If no occurance is found, then a -1
is returned.
So, while it's more complicated, we can look for our \n
character, and then look at the next slice of the array after that character, and then look at the next slice... continuing on until we're out of things to look at. Then we'd move on to the next chunk of file.
In that implementation, we would replace count += bytes.Count(buffer[:read], target)
with the following
...
const target byte = '\n'
...
var position int
for {
idxOf := bytes.IndexByte(buffer[position:read], target)
if idxOf == -1 {
break
}
count++
position += idxOf + 1
}
Is it Fast?
I am only concerned with if this is comparatively fast when measured against wc
. Getting true benchmarking numbers are outside of the scope of my efforts here. I've run both programs several times which ensures the operating system and my storage have both done any caching they plan to do.
Using time
(Unix timing utility) wc
on my machine (a midrange dev laptop with an NVMe SSD drive) to parse a 1.6GB text file of lorem ipsum text, I get the following averages after an initial warmup call:
real 0m0.822s
user 0m0.156s
sys 0m0.655s
Using lc
to parse the same file, I get the following averages after an initial warmup call:
real 0m0.625s
user 0m0.015s
sys 0m0.015s
So, I'm happy with how this experiment went.
Wrapping Up?
I haven't demonstrated any tests for this program. I'll leave you to review them at your leisure. Or, wait for the next exciting installment.
I've previously covered how I set Go up locally and added %GOPATH%\bin
to my path. So, from within the lc
project directory, I can run go install
and have a shiny new command line utility to use.
Honestly, thinking up bespoke little utilities has been a lot of fun. And, once you unlock the power of chaining them together, you'll think of many new use cases. Just keep the Unix philosophy in mind.
Feel free to ping me on Bluesky @hyrmn with any questions or comments.