Skip to content

Commit

Permalink
Merge pull request #98 from posener/refactor
Browse files Browse the repository at this point in the history
Some refactorings
  • Loading branch information
posener authored Jul 5, 2019
2 parents 6ffe496 + 72c5c94 commit 2f2ff27
Show file tree
Hide file tree
Showing 11 changed files with 217 additions and 101 deletions.
16 changes: 8 additions & 8 deletions .travis.yml
Original file line number Diff line number Diff line change
@@ -1,16 +1,16 @@
language: go
sudo: false
go:
- 1.11
- tip
- 1.12.x
- 1.11.x
- 1.10.x
- 1.9
- 1.8

before_install:
- go get -u -t ./...

script:
- GO111MODULE=on ./test.sh
- go test -race -coverprofile=coverage.txt -covermode=atomic ./...

after_success:
- bash <(curl -s https://codecov.io/bash)

matrix:
allow_failures:
- go: tip
2 changes: 2 additions & 0 deletions args.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ type Args struct {
// Directory gives the directory of the current written
// last argument if it represents a file name being written.
// in case that it is not, we fall back to the current directory.
//
// Deprecated.
func (a Args) Directory() string {
if info, err := os.Stat(a.Last); err == nil && info.IsDir() {
return fixPathForm(a.Last, a.Last)
Expand Down
4 changes: 2 additions & 2 deletions complete.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,9 @@ import (
"io"
"os"
"strconv"
"strings"

"github.com/posener/complete/cmd"
"github.com/posener/complete/match"
)

const (
Expand Down Expand Up @@ -72,7 +72,7 @@ func (c *Complete) Complete() bool {
// filter only options that match the last argument
matches := []string{}
for _, option := range options {
if match.Prefix(option, a.Last) {
if strings.HasPrefix(option, a.Last) {
matches = append(matches, option)
}
}
Expand Down
2 changes: 1 addition & 1 deletion gocomplete/complete.go
Original file line number Diff line number Diff line change
Expand Up @@ -449,7 +449,7 @@ func main() {
"-unusedfuncs": complete.PredictAnything,
"-unusedresult": complete.PredictNothing,
"-unusedstringmethods": complete.PredictAnything,
"-v": complete.PredictNothing,
"-v": complete.PredictNothing,
},
Args: anyGo,
},
Expand Down
19 changes: 0 additions & 19 deletions match/file.go

This file was deleted.

32 changes: 32 additions & 0 deletions match/match.go
Original file line number Diff line number Diff line change
@@ -1,7 +1,39 @@
// Package match contains matchers that decide if to apply completion.
//
// This package is deprecated.
package match

import "strings"

// Match matches two strings
// it is used for comparing a term to the last typed
// word, the prefix, and see if it is a possible auto complete option.
//
// Deprecated.
type Match func(term, prefix string) bool

// Prefix is a simple Matcher, if the word is it's prefix, there is a match
// Match returns true if a has the prefix as prefix
//
// Deprecated.
func Prefix(long, prefix string) bool {
return strings.HasPrefix(long, prefix)
}

// File returns true if prefix can match the file
//
// Deprecated.
func File(file, prefix string) bool {
// special case for current directory completion
if file == "./" && (prefix == "." || prefix == "") {
return true
}
if prefix == "." && strings.HasPrefix(file, ".") {
return true
}

file = strings.TrimPrefix(file, "./")
prefix = strings.TrimPrefix(prefix, "./")

return strings.HasPrefix(file, prefix)
}
9 changes: 0 additions & 9 deletions match/prefix.go

This file was deleted.

74 changes: 70 additions & 4 deletions predict_files.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,6 @@ import (
"os"
"path/filepath"
"strings"

"github.com/posener/complete/match"
)

// PredictDirs will search for directories in the given started to be typed
Expand Down Expand Up @@ -53,7 +51,7 @@ func predictFiles(a Args, pattern string, allowFiles bool) []string {
return nil
}

dir := a.Directory()
dir := directory(a.Last)
files := listFiles(dir, pattern, allowFiles)

// add dir if match
Expand All @@ -62,6 +60,19 @@ func predictFiles(a Args, pattern string, allowFiles bool) []string {
return PredictFilesSet(files).Predict(a)
}

// directory gives the directory of the given partial path
// in case that it is not, we fall back to the current directory.
func directory(path string) string {
if info, err := os.Stat(path); err == nil && info.IsDir() {
return fixPathForm(path, path)
}
dir := filepath.Dir(path)
if info, err := os.Stat(dir); err == nil && info.IsDir() {
return fixPathForm(path, dir)
}
return "./"
}

// PredictFilesSet predict according to file rules to a given set of file names
func PredictFilesSet(files []string) PredictFunc {
return func(a Args) (prediction []string) {
Expand All @@ -70,7 +81,7 @@ func PredictFilesSet(files []string) PredictFunc {
f = fixPathForm(a.Last, f)

// test matching of file to the argument
if match.File(f, a.Last) {
if matchFile(f, a.Last) {
prediction = append(prediction, f)
}
}
Expand Down Expand Up @@ -106,3 +117,58 @@ func listFiles(dir, pattern string, allowFiles bool) []string {
}
return list
}

// MatchFile returns true if prefix can match the file
func matchFile(file, prefix string) bool {
// special case for current directory completion
if file == "./" && (prefix == "." || prefix == "") {
return true
}
if prefix == "." && strings.HasPrefix(file, ".") {
return true
}

file = strings.TrimPrefix(file, "./")
prefix = strings.TrimPrefix(prefix, "./")

return strings.HasPrefix(file, prefix)
}

// fixPathForm changes a file name to a relative name
func fixPathForm(last string, file string) string {
// get wording directory for relative name
workDir, err := os.Getwd()
if err != nil {
return file
}

abs, err := filepath.Abs(file)
if err != nil {
return file
}

// if last is absolute, return path as absolute
if filepath.IsAbs(last) {
return fixDirPath(abs)
}

rel, err := filepath.Rel(workDir, abs)
if err != nil {
return file
}

// fix ./ prefix of path
if rel != "." && strings.HasPrefix(last, ".") {
rel = "./" + rel
}

return fixDirPath(rel)
}

func fixDirPath(path string) string {
info, err := os.Stat(path)
if err == nil && info.IsDir() && !strings.HasSuffix(path, "/") {
path += "/"
}
return path
}
102 changes: 102 additions & 0 deletions predict_test.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
package complete

import (
"fmt"
"os"
"sort"
"strings"
"testing"
Expand Down Expand Up @@ -167,3 +169,103 @@ func TestPredicate(t *testing.T) {
}
}
}

func TestMatchFile(t *testing.T) {
t.Parallel()

// Change to tests directory for testing completion of
// files and directories
err := os.Chdir("../tests")
if err != nil {
panic(err)
}

type matcherTest struct {
prefix string
want bool
}

tests := []struct {
long string
tests []matcherTest
}{
{
long: "file.txt",
tests: []matcherTest{
{prefix: "", want: true},
{prefix: "f", want: true},
{prefix: "./f", want: true},
{prefix: "./.", want: false},
{prefix: "file.", want: true},
{prefix: "./file.", want: true},
{prefix: "file.txt", want: true},
{prefix: "./file.txt", want: true},
{prefix: "other.txt", want: false},
{prefix: "/other.txt", want: false},
{prefix: "/file.txt", want: false},
{prefix: "/fil", want: false},
{prefix: "/file.txt2", want: false},
{prefix: "/.", want: false},
},
},
{
long: "./file.txt",
tests: []matcherTest{
{prefix: "", want: true},
{prefix: "f", want: true},
{prefix: "./f", want: true},
{prefix: "./.", want: false},
{prefix: "file.", want: true},
{prefix: "./file.", want: true},
{prefix: "file.txt", want: true},
{prefix: "./file.txt", want: true},
{prefix: "other.txt", want: false},
{prefix: "/other.txt", want: false},
{prefix: "/file.txt", want: false},
{prefix: "/fil", want: false},
{prefix: "/file.txt2", want: false},
{prefix: "/.", want: false},
},
},
{
long: "/file.txt",
tests: []matcherTest{
{prefix: "", want: true},
{prefix: "f", want: false},
{prefix: "./f", want: false},
{prefix: "./.", want: false},
{prefix: "file.", want: false},
{prefix: "./file.", want: false},
{prefix: "file.txt", want: false},
{prefix: "./file.txt", want: false},
{prefix: "other.txt", want: false},
{prefix: "/other.txt", want: false},
{prefix: "/file.txt", want: true},
{prefix: "/fil", want: true},
{prefix: "/file.txt2", want: false},
{prefix: "/.", want: false},
},
},
{
long: "./",
tests: []matcherTest{
{prefix: "", want: true},
{prefix: ".", want: true},
{prefix: "./", want: true},
{prefix: "./.", want: false},
},
},
}

for _, tt := range tests {
for _, ttt := range tt.tests {
name := fmt.Sprintf("long=%q&prefix=%q", tt.long, ttt.prefix)
t.Run(name, func(t *testing.T) {
got := matchFile(tt.long, ttt.prefix)
if got != ttt.want {
t.Errorf("Failed %s: got = %t, want: %t", name, got, ttt.want)
}
})
}
}
}
12 changes: 0 additions & 12 deletions test.sh

This file was deleted.

Loading

0 comments on commit 2f2ff27

Please sign in to comment.