mirror of https://github.com/junegunn/fzf
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
118 lines
2.6 KiB
Go
118 lines
2.6 KiB
Go
package fzf
|
|
|
|
import (
|
|
"bufio"
|
|
"io"
|
|
"os"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/junegunn/fzf/src/util"
|
|
)
|
|
|
|
// Reader reads from command or standard input
|
|
type Reader struct {
|
|
pusher func([]byte) bool
|
|
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)
|
|
}
|
|
|
|
// ReadSource reads data from the default command or from standard input
|
|
func (r *Reader) ReadSource() {
|
|
r.startEventPoller()
|
|
var success bool
|
|
if util.IsTty() {
|
|
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)
|
|
}
|
|
} else {
|
|
success = r.readFromStdin()
|
|
}
|
|
r.fin(success)
|
|
}
|
|
|
|
func (r *Reader) feed(src io.Reader) {
|
|
delim := byte('\n')
|
|
if r.delimNil {
|
|
delim = '\000'
|
|
}
|
|
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)
|
|
byteaLen := len(bytea)
|
|
if byteaLen > 0 {
|
|
if err == nil {
|
|
// get rid of carriage return if under Windows:
|
|
if util.IsWindows() && byteaLen >= 2 && bytea[byteaLen-2] == byte('\r') {
|
|
bytea = bytea[:byteaLen-2]
|
|
} else {
|
|
bytea = bytea[:byteaLen-1]
|
|
}
|
|
}
|
|
if r.pusher(bytea) {
|
|
atomic.StoreInt32(&r.event, int32(EvtReadNew))
|
|
}
|
|
}
|
|
if err != nil {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
func (r *Reader) readFromStdin() bool {
|
|
r.feed(os.Stdin)
|
|
return true
|
|
}
|
|
|
|
func (r *Reader) readFromCommand(shell string, cmd string) bool {
|
|
listCommand := util.ExecCommandWith(shell, cmd)
|
|
out, err := listCommand.StdoutPipe()
|
|
if err != nil {
|
|
return false
|
|
}
|
|
err = listCommand.Start()
|
|
if err != nil {
|
|
return false
|
|
}
|
|
r.feed(out)
|
|
return listCommand.Wait() == nil
|
|
}
|