fzf/src/reader.go

118 lines
2.6 KiB
Go
Raw Normal View History

2015-01-01 19:49:30 +00:00
package fzf
import (
"bufio"
"io"
"os"
"sync/atomic"
"time"
2015-01-12 03:56:17 +00:00
"github.com/junegunn/fzf/src/util"
2015-01-01 19:49:30 +00:00
)
2015-01-11 18:01:24 +00:00
// Reader reads from command or standard input
2015-01-01 19:49:30 +00:00
type Reader struct {
pusher func([]byte) bool
2015-01-12 03:56:17 +00:00
eventBox *util.EventBox
delimNil bool
event int32
}
// NewReader returns new Reader object
func NewReader(pusher func([]byte) bool, eventBox *util.EventBox, delimNil bool) *Reader {
return &Reader{pusher, eventBox, delimNil, int32(EvtReady)}
}
func (r *Reader) startEventPoller() {
go func() {
ptr := &r.event
pollInterval := readerPollIntervalMin
for {
if atomic.CompareAndSwapInt32(ptr, int32(EvtReadNew), int32(EvtReady)) {
r.eventBox.Set(EvtReadNew, true)
pollInterval = readerPollIntervalMin
} else if atomic.LoadInt32(ptr) == int32(EvtReadFin) {
return
} else {
pollInterval += readerPollIntervalStep
if pollInterval > readerPollIntervalMax {
pollInterval = readerPollIntervalMax
}
}
time.Sleep(pollInterval)
}
}()
}
func (r *Reader) fin(success bool) {
atomic.StoreInt32(&r.event, int32(EvtReadFin))
r.eventBox.Set(EvtReadFin, success)
2015-01-01 19:49:30 +00:00
}
2015-01-11 18:01:24 +00:00
// ReadSource reads data from the default command or from standard input
2015-01-01 19:49:30 +00:00
func (r *Reader) ReadSource() {
r.startEventPoller()
var success bool
2015-01-12 03:56:17 +00:00
if util.IsTty() {
2015-01-01 19:49:30 +00:00
cmd := os.Getenv("FZF_DEFAULT_COMMAND")
if len(cmd) == 0 {
// The default command for *nix requires bash
success = r.readFromCommand("bash", defaultCommand)
} else {
success = r.readFromCommand("sh", cmd)
2015-01-01 19:49:30 +00:00
}
} else {
success = r.readFromStdin()
2015-01-01 19:49:30 +00:00
}
r.fin(success)
2015-01-01 19:49:30 +00:00
}
func (r *Reader) feed(src io.Reader) {
delim := byte('\n')
if r.delimNil {
delim = '\000'
}
2016-08-15 16:52:24 +00:00
reader := bufio.NewReaderSize(src, readerBufferSize)
for {
// ReadBytes returns err != nil if and only if the returned data does not
// end in delim.
bytea, err := reader.ReadBytes(delim)
2016-10-24 03:45:45 +00:00
byteaLen := len(bytea)
2017-08-25 18:24:42 +00:00
if byteaLen > 0 {
if err == nil {
2016-10-24 03:45:45 +00:00
// get rid of carriage return if under Windows:
if util.IsWindows() && byteaLen >= 2 && bytea[byteaLen-2] == byte('\r') {
2016-10-24 03:45:45 +00:00
bytea = bytea[:byteaLen-2]
} else {
bytea = bytea[:byteaLen-1]
}
}
if r.pusher(bytea) {
atomic.StoreInt32(&r.event, int32(EvtReadNew))
2015-07-21 18:21:20 +00:00
}
2015-01-01 19:49:30 +00:00
}
if err != nil {
break
}
2015-01-01 19:49:30 +00:00
}
}
func (r *Reader) readFromStdin() bool {
2015-01-01 19:49:30 +00:00
r.feed(os.Stdin)
return true
2015-01-01 19:49:30 +00:00
}
func (r *Reader) readFromCommand(shell string, cmd string) bool {
listCommand := util.ExecCommandWith(shell, cmd, false)
2015-01-01 19:49:30 +00:00
out, err := listCommand.StdoutPipe()
if err != nil {
return false
2015-01-01 19:49:30 +00:00
}
err = listCommand.Start()
if err != nil {
return false
2015-01-01 19:49:30 +00:00
}
r.feed(out)
return listCommand.Wait() == nil
2015-01-01 19:49:30 +00:00
}