Refactoring for test

This commit is contained in:
Junegunn Choi 2013-11-14 20:50:27 +09:00
parent 67bdc3a0ad
commit 90ad6d50b8
3 changed files with 796 additions and 584 deletions

336
fzf
View File

@ -10,7 +10,7 @@
# URL: https://github.com/junegunn/fzf # URL: https://github.com/junegunn/fzf
# Author: Junegunn Choi # Author: Junegunn Choi
# License: MIT # License: MIT
# Last update: November 10, 2013 # Last update: November 15, 2013
# #
# Copyright (c) 2013 Junegunn Choi # Copyright (c) 2013 Junegunn Choi
# #
@ -35,8 +35,78 @@
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
require 'thread'
require 'curses'
class FZF
C = Curses
attr_reader :rxflag, :sort, :color, :multi
class AtomicVar
def initialize value
@value = value
@mutex = Mutex.new
end
def get
@mutex.synchronize { @value }
end
def set value = nil
if block_given?
@mutex.synchronize { @value = yield @value }
else
@mutex.synchronize { @value = value }
end
end
def method_missing sym, *args, &blk
@mutex.synchronize { @value.send(sym, *args, &blk) }
end
end
def initialize argv, source = $stdin
usage 0 unless (%w[--help -h] & argv).empty?
@rxflag = argv.delete('+i') ? 0 : Regexp::IGNORECASE
@sort = %w[+s --no-sort].map { |e| argv.delete e }.compact.empty? ?
ENV.fetch('FZF_DEFAULT_SORT', 500).to_i : nil
@color = %w[+c --no-color].map { |e| argv.delete e }.compact.empty?
@multi = !%w[-m --multi].map { |e| argv.delete e }.compact.empty?
rest = argv.join ' '
if sort = rest.match(/(-s|--sort=?) ?([0-9]+)/)
usage 1 unless @sort
@sort = sort[2].to_i
rest = rest.delete sort[0]
end
usage 1 unless rest.empty?
@source = source
@mtx = Mutex.new
@cv = ConditionVariable.new
@events = {}
@new = []
@smtx = Mutex.new
@cursor_x = AtomicVar.new(0)
@query = AtomicVar.new('')
@matches = AtomicVar.new([])
@count = AtomicVar.new(0)
@vcursor = AtomicVar.new(0)
@fan = AtomicVar.new('-\|/-\|/'.split(//))
@selects = AtomicVar.new({}) # ordered >= 1.9
end
def start
@stdout = $stdout.clone
$stdout.reopen($stderr)
init_screen
start_reader
start_search
start_loop
end
def usage x def usage x
puts %[usage: fzf [options] $stderr.puts %[usage: fzf [options]
-m, --multi Enable multi-select -m, --multi Enable multi-select
-s, --sort=MAX Maximum number of matched items to sort. Default: 500. -s, --sort=MAX Maximum number of matched items to sort. Default: 500.
@ -46,39 +116,6 @@ def usage x
exit x exit x
end end
stdout = $stdout.clone
$stdout.reopen($stderr)
usage 0 unless (%w[--help -h] & ARGV).empty?
@rxflag = ARGV.delete('+i') ? 0 : Regexp::IGNORECASE
@sort = %w[+s --no-sort].map { |e| ARGV.delete e }.compact.empty? ?
ENV.fetch('FZF_DEFAULT_SORT', 500).to_i : nil
@color = %w[+c --no-color].map { |e| ARGV.delete e }.compact.empty?
@multi = !%w[-m --multi].map { |e| ARGV.delete e }.compact.empty?
rest = ARGV.join ' '
if sort = rest.match(/(-s|--sort=?) ?([0-9]+)/)
usage 1 unless @sort
@sort = sort[2].to_i
rest = rest.delete sort[0]
end
usage 1 unless rest.empty?
require 'thread'
require 'curses'
@mtx = Mutex.new
@smtx = Mutex.new
@cv = ConditionVariable.new
@lists = []
@new = []
@query = ''
@matches = []
@count = 0
@cursor_x = 0
@vcursor = 0
@events = {}
@selects = {} # ordered >= 1.9
case RUBY_PLATFORM case RUBY_PLATFORM
when /darwin/ when /darwin/
module UConv module UConv
@ -145,20 +182,24 @@ when /darwin/
end end
end end
def convert_query q
UConv.nfd(q).split(//)
end
def convert_item item def convert_item item
UConv.nfc(*item) UConv.nfc(*item)
end end
class Matcher
def convert_query q
UConv.nfd(q).split(//)
end
end
else else
def convert_item item
item
end
class Matcher
def convert_query q def convert_query q
q.split(//) q.split(//)
end end
def convert_item item
item
end end
end end
@ -169,7 +210,6 @@ def emit event
end end
end end
C = Curses
def max_items; C.lines - 2; end def max_items; C.lines - 2; end
def cursor_y; C.lines - 1; end def cursor_y; C.lines - 1; end
def cprint str, col def cprint str, col
@ -183,12 +223,11 @@ def print_input
C.clrtoeol C.clrtoeol
cprint '> ', color(:blue, true) cprint '> ', color(:blue, true)
C.attron(C::A_BOLD) do C.attron(C::A_BOLD) do
C.addstr @query C.addstr @query.get
end end
end end
def print_info selected, msg = nil def print_info msg = nil
@fan ||= '-\|/-\|/'.split(//)
C.setpos cursor_y - 1, 0 C.setpos cursor_y - 1, 0
C.clrtoeol C.clrtoeol
prefix = prefix =
@ -200,14 +239,16 @@ def print_info selected, msg = nil
' ' ' '
end end
C.attron color(:info, false) do C.attron color(:info, false) do
C.addstr "#{prefix}#{@matches.length}/#{@count}" C.addstr "#{prefix}#{@matches.length}/#{@count.get}"
C.addstr " (#{selected})" if selected > 0 if (selected = @selects.length) > 0
C.addstr " (#{selected})"
end
C.addstr msg if msg C.addstr msg if msg
end end
end end
def refresh def refresh
C.setpos cursor_y, 2 + width(@query[0, @cursor_x]) C.setpos cursor_y, 2 + width(@query[0, @cursor_x.get])
C.refresh C.refresh
end end
@ -278,17 +319,26 @@ def print_item row, tokens, chosen, selected
C.attroff color(:chosen, true) if chosen C.attroff color(:chosen, true) if chosen
end end
def sort_by_rank list
list.sort_by { |tuple|
line, offsets = tuple
matchlen = (offsets.map { |pair| pair.last }.max || 0) -
(offsets.map { |pair| pair.first }.min || 0)
[matchlen, line.length, line]
}
end
if RUBY_VERSION.split('.').map { |e| e.rjust(3, '0') }.join > '001009' if RUBY_VERSION.split('.').map { |e| e.rjust(3, '0') }.join > '001009'
@wrx = Regexp.new '\p{Han}|\p{Katakana}|\p{Hiragana}|\p{Hangul}' @@wrx = Regexp.new '\p{Han}|\p{Katakana}|\p{Hiragana}|\p{Hangul}'
def width str def width str
str.gsub(@wrx, ' ').length str.gsub(@@wrx, ' ').length
end end
def trim str, len, left def trim str, len, left
width = width str width = width str
diff = 0 diff = 0
while width > len while width > len
width -= (left ? str[0, 1] : str[-1, 1]) =~ @wrx ? 2 : 1 width -= (left ? str[0, 1] : str[-1, 1]) =~ @@wrx ? 2 : 1
str = left ? str[1..-1] : str[0...-1] str = left ? str[1..-1] : str[0...-1]
diff += 1 diff += 1
end end
@ -308,19 +358,20 @@ else
end end
end end
class String class ::String
def ord def ord
self.unpack('c').first self.unpack('c').first
end end
end end
class Fixnum class ::Fixnum
def ord def ord
self self
end end
end end
end end
def init_screen
C.init_screen C.init_screen
C.start_color C.start_color
dbg = dbg =
@ -354,13 +405,13 @@ if @color
C.init_pair 7, C::COLOR_RED, C::COLOR_BLACK C.init_pair 7, C::COLOR_RED, C::COLOR_BLACK
end end
def color sym, bold = false def self.color sym, bold = false
C.color_pair([:blue, :match, :chosen, C.color_pair([:blue, :match, :chosen,
:match!, :fan, :info, :red].index(sym) + 1) | :match!, :fan, :info, :red].index(sym) + 1) |
(bold ? C::A_BOLD : 0) (bold ? C::A_BOLD : 0)
end end
else else
def color sym, bold = false def self.color sym, bold = false
case sym case sym
when :chosen when :chosen
bold ? C::A_REVERSE : 0 bold ? C::A_REVERSE : 0
@ -373,9 +424,11 @@ else
end | (bold ? C::A_BOLD : 0) end | (bold ? C::A_BOLD : 0)
end end
end end
end
@read = def start_reader
if $stdin.tty? stream =
if @source.tty?
if default_command = ENV['FZF_DEFAULT_COMMAND'] if default_command = ENV['FZF_DEFAULT_COMMAND']
IO.popen(default_command) IO.popen(default_command)
elsif !`which find`.empty? elsif !`which find`.empty?
@ -384,27 +437,28 @@ end
exit 1 exit 1
end end
else else
$stdin @source
end end
reader = Thread.new { Thread.new do
while line = @read.gets while line = stream.gets
emit(:new) { @new << line.chomp } emit(:new) { @new << line.chomp }
end end
emit(:loaded) { true } emit(:loaded) { true }
@smtx.synchronize { @fan = [] } @fan.clear
} end
end
def start_search
main = Thread.current main = Thread.current
matcher = FuzzyMatcher.new @rxflag
searcher = Thread.new { searcher = Thread.new {
lists = []
events = {} events = {}
fcache = {} fcache = {}
matches = []
selects = {}
mcount = 0 # match count mcount = 0 # match count
plcount = 0 # prev list count plcount = 0 # prev list count
q = '' q = ''
vcursor = 0
delay = -5 delay = -5
begin begin
@ -422,15 +476,11 @@ searcher = Thread.new {
end end
if events[:new] if events[:new]
@lists << [@new, {}] lists << @new
@count += @new.length @count.set { |c| c + @new.length }
@new = [] @new = []
fcache = {} fcache = {}
end end
if events[:select]
selects = @selects.dup
end
end#mtx end#mtx
new_search = events[:key] || events.delete(:new) new_search = events[:key] || events.delete(:new)
@ -438,71 +488,38 @@ searcher = Thread.new {
progress = 0 progress = 0
started_at = Time.now started_at = Time.now
if new_search && !@lists.empty? if new_search && !lists.empty?
q = events.delete(:key) || q q, cx = events.delete(:key) || [q, 0]
unless q.empty? plcount = [@matches.length, max_items].min
q = q.downcase if @rxflag != 0 @matches.set(fcache[q] ||=
regexp = Regexp.new(convert_query(q).inject('') { |sum, e|
e = Regexp.escape e
sum << "#{e}[^#{e}]*?"
}, @rxflag)
end
matches = fcache[q] ||=
begin begin
found = [] found = []
skip = false skip = false
cnt = 0 cnt = 0
@lists.each do |pair| lists.each do |list|
list, cache = pair
cnt += list.length cnt += list.length
skip = @mtx.synchronize { @events[:key] }
@mtx.synchronize {
skip = @events[:key]
progress = (100 * cnt / @count)
}
break if skip break if skip
found.concat(cache[q] ||= q.empty? ? list : begin progress = (100 * cnt / @count.get)
if progress < 100 && Time.now - started_at > 0.5 if progress < 100 && Time.now - started_at > 0.5 && !q.empty?
@smtx.synchronize do @smtx.synchronize do
print_info selects.length, " (#{progress}%)" print_info " (#{progress}%)"
refresh refresh
end end
end end
prefix, suffix = @query[0, @cursor_x], @query[@cursor_x..-1] || '' found.concat(q.empty? ? list :
prefix_cache = suffix_cache = nil matcher.match(list, q, q[0, cx], q[cx..-1]))
(prefix.length - 1).downto(1) do |len|
break if prefix_cache = cache[prefix[0, len]]
end
0.upto(suffix.length - 1) do |idx|
break if suffix_cache = cache[suffix[idx..-1]]
end unless suffix.empty?
partial_cache = [prefix_cache, suffix_cache].compact.sort_by { |e| e.length }.first
(partial_cache ? partial_cache.map { |e| e.first } : list).map { |line|
# Ignore errors: e.g. invalid byte sequence in UTF-8
md = line.match(regexp) rescue nil
md && [line, [md.offset(0)]]
}.compact
end)
end end
next if skip next if skip
@sort ? found : found.reverse @sort ? found : found.reverse
end end)
mcount = matches.length mcount = @matches.length
if @sort && mcount <= @sort && !q.empty? if @sort && mcount <= @sort && !q.empty?
matches.replace matches.sort_by { |tuple| @matches.set { |m| sort_by_rank m }
line, offsets = tuple
matchlen = offsets.map { |pair| pair.last }.max || 0 -
offsets.map { |pair| pair.first }.min || 0
[matchlen, line.length, line]
}
end end
end#new_search end#new_search
@ -510,11 +527,7 @@ searcher = Thread.new {
sleep((delay = [20, delay + 5].min) * 0.01) unless user_input sleep((delay = [20, delay + 5].min) * 0.01) unless user_input
if events.delete(:vcursor) || new_search if events.delete(:vcursor) || new_search
@mtx.synchronize do @vcursor.set { |v| [0, [v, mcount - 1, max_items - 1].min].max }
plcount = [@matches.length, max_items].min
@matches = matches
vcursor = @vcursor = [0, [@vcursor, mcount - 1, max_items - 1].min].max
end
end end
# Output # Output
@ -528,17 +541,18 @@ searcher = Thread.new {
end end
maxc = C.cols - 3 maxc = C.cols - 3
matches[0, max_items].each_with_index do |item, idx| vcursor = @vcursor.get
@matches[0, max_items].each_with_index do |item, idx|
next if !new_search && !((vcursor-1)..(vcursor+1)).include?(idx) next if !new_search && !((vcursor-1)..(vcursor+1)).include?(idx)
row = cursor_y - idx - 2 row = cursor_y - idx - 2
chosen = idx == vcursor chosen = idx == vcursor
selected = selects.include?([*item][0]) selected = @selects.include?([*item][0])
line, offsets = convert_item item line, offsets = convert_item item
tokens = format line, maxc, offsets tokens = format line, maxc, offsets
print_item row, tokens, chosen, selected print_item row, tokens, chosen, selected
end end
print_info selects.length if !@lists.empty? || events[:loaded] print_info if !lists.empty? || events[:loaded]
refresh refresh
end end
end#while end#while
@ -546,7 +560,9 @@ searcher = Thread.new {
main.raise e main.raise e
end end
} }
end
def start_loop
got = nil got = nil
begin begin
tty = IO.open(IO.sysopen('/dev/tty'), 'r') tty = IO.open(IO.sysopen('/dev/tty'), 'r')
@ -557,16 +573,14 @@ begin
ctrl(:c) => proc { exit 1 }, ctrl(:c) => proc { exit 1 },
ctrl(:d) => proc { exit 1 if input.empty? }, ctrl(:d) => proc { exit 1 if input.empty? },
ctrl(:m) => proc { ctrl(:m) => proc {
@mtx.synchronize do got = [*@matches.fetch(@vcursor.get, [])][0]
got = [*@matches.fetch(@vcursor, [])][0]
end
exit 0 exit 0
}, },
ctrl(:u) => proc { input = input[cursor..-1]; cursor = 0 }, ctrl(:u) => proc { input = input[cursor..-1]; cursor = 0 },
ctrl(:a) => proc { cursor = 0 }, ctrl(:a) => proc { cursor = 0 },
ctrl(:e) => proc { cursor = input.length }, ctrl(:e) => proc { cursor = input.length },
ctrl(:j) => proc { emit(:vcursor) { @vcursor -= 1 } }, ctrl(:j) => proc { emit(:vcursor) { @vcursor.set { |v| v - 1 } } },
ctrl(:k) => proc { emit(:vcursor) { @vcursor += 1 } }, ctrl(:k) => proc { emit(:vcursor) { @vcursor.set { |v| v + 1 } } },
ctrl(:w) => proc { ctrl(:w) => proc {
ridx = (input[0...cursor - 1].rindex(/\S\s/) || -2) + 2 ridx = (input[0...cursor - 1].rindex(/\S\s/) || -2) + 2
input = input[0...ridx] + input[cursor..-1] input = input[0...ridx] + input[cursor..-1]
@ -575,13 +589,13 @@ begin
127 => proc { input[cursor -= 1] = '' if cursor > 0 }, 127 => proc { input[cursor -= 1] = '' if cursor > 0 },
9 => proc { |o| 9 => proc { |o|
emit(:select) { emit(:select) {
if sel = [*@matches.fetch(@vcursor, [])][0] if sel = [*@matches.fetch(@vcursor.get, [])][0]
if @selects.has_key? sel if @selects.has_key? sel
@selects.delete sel @selects.delete sel
else else
@selects[sel] = 1 @selects[sel] = 1
end end
@vcursor = [0, @vcursor + (o == :stab ? 1 : -1)].max @vcursor.set { |v| [0, v + (o == :stab ? 1 : -1)].max }
end end
} if @multi } if @multi
}, },
@ -596,9 +610,9 @@ begin
actions[:stab] = actions[9] actions[:stab] = actions[9]
while true while true
@cursor_x.set cursor
# Update user input # Update user input
@smtx.synchronize do @smtx.synchronize do
@cursor_x = cursor
print_input print_input
refresh refresh
end end
@ -627,16 +641,62 @@ begin
}).call(ord) }).call(ord)
# Dispatch key event # Dispatch key event
emit(:key) { @query = input.dup } emit(:key) { [@query.set(input.dup), cursor] }
end end
ensure ensure
C.close_screen C.close_screen
if got if got
@selects.delete got @selects.delete got
@selects.each do |sel, _| @selects.each do |sel, _|
stdout.puts sel @stdout.puts sel
end end
stdout.puts got @stdout.puts got
end
exit 0
end end
end end
class FuzzyMatcher < Matcher
attr_reader :cache
def initialize rxflag
@cache = Hash.new { |h, k| h[k] = {} }
@regexp = {}
@rxflag = rxflag
end
def match list, q, prefix, suffix
regexp = @regexp[q] ||= begin
q = q.downcase if @rxflag != 0
Regexp.new(convert_query(q).inject('') { |sum, e|
e = Regexp.escape e
sum << "#{e}[^#{e}]*?"
}, @rxflag)
end
cache = @cache[list.object_id]
prefix_cache = nil
(prefix.length - 1).downto(1) do |len|
break if prefix_cache = cache[prefix[0, len]]
end
suffix_cache = nil
0.upto(suffix.length - 1) do |idx|
break if suffix_cache = cache[suffix[idx..-1]]
end unless suffix.empty?
partial_cache = [prefix_cache,
suffix_cache].compact.sort_by { |e| e.length }.first
cache[q] ||= (partial_cache ?
partial_cache.map { |e| e.first } : list).map { |line|
# Ignore errors: e.g. invalid byte sequence in UTF-8
md = line.match(regexp) rescue nil
md && [line, [md.offset(0)]]
}.compact
end
end
end#FZF
FZF.new(ARGV, $stdin).start if $0 == __FILE__

View File

@ -1,7 +1,7 @@
# coding: utf-8 # coding: utf-8
Gem::Specification.new do |spec| Gem::Specification.new do |spec|
spec.name = 'fzf' spec.name = 'fzf'
spec.version = '0.3.1' spec.version = '0.4.0'
spec.authors = ['Junegunn Choi'] spec.authors = ['Junegunn Choi']
spec.email = ['junegunn.c@gmail.com'] spec.email = ['junegunn.c@gmail.com']
spec.description = %q{Fuzzy finder for your shell} spec.description = %q{Fuzzy finder for your shell}

152
test/test_fzf.rb Normal file
View File

@ -0,0 +1,152 @@
#!/usr/bin/env ruby
# encoding: utf-8
require 'minitest/autorun'
$LOAD_PATH.unshift File.expand_path('../..', __FILE__)
load 'fzf'
class TestFZF < MiniTest::Unit::TestCase
def test_default_options
fzf = FZF.new []
assert_equal 500, fzf.sort
assert_equal false, fzf.multi
assert_equal true, fzf.color
assert_equal Regexp::IGNORECASE, fzf.rxflag
begin
ENV['FZF_DEFAULT_SORT'] = '1000'
fzf = FZF.new []
assert_equal 1000, fzf.sort
ensure
ENV.delete 'FZF_DEFAULT_SORT'
end
end
def test_option_parser
# Long opts
fzf = FZF.new %w[--sort=2000 --no-color --multi +i]
assert_equal 2000, fzf.sort
assert_equal true, fzf.multi
assert_equal false, fzf.color
assert_equal 0, fzf.rxflag
# Short opts
fzf = FZF.new %w[-s 2000 +c -m +i]
assert_equal 2000, fzf.sort
assert_equal true, fzf.multi
assert_equal false, fzf.color
assert_equal 0, fzf.rxflag
end
def test_invalid_option
[%w[-s 2000 +s], %w[yo dawg]].each do |argv|
assert_raises(SystemExit) do
fzf = FZF.new argv
end
end
end
# FIXME Only on 1.9 or above
def test_width
fzf = FZF.new []
assert_equal 5, fzf.width('abcde')
assert_equal 4, fzf.width('한글')
assert_equal 5, fzf.width('한글.')
end
def test_trim
fzf = FZF.new []
assert_equal ['사.', 6], fzf.trim('가나다라마바사.', 4, true)
assert_equal ['바사.', 5], fzf.trim('가나다라마바사.', 5, true)
assert_equal ['가나', 6], fzf.trim('가나다라마바사.', 4, false)
assert_equal ['가나', 6], fzf.trim('가나다라마바사.', 5, false)
assert_equal ['가나a', 6], fzf.trim('가나ab라마바사.', 5, false)
end
def test_format
fzf = FZF.new []
assert_equal [['01234..', false]], fzf.format('0123456789', 7, [])
assert_equal [['012', false], ['34', true], ['..', false]],
fzf.format('0123456789', 7, [[3, 5]])
assert_equal [['..56', false], ['789', true]],
fzf.format('0123456789', 7, [[7, 10]])
assert_equal [['..56', false], ['78', true], ['9', false]],
fzf.format('0123456789', 7, [[7, 9]])
(3..5).each do |i|
assert_equal [['..', false], ['567', true], ['89', false]],
fzf.format('0123456789', 7, [[i, 8]])
end
assert_equal [['..', false], ['345', true], ['..', false]],
fzf.format('0123456789', 7, [[3, 6]])
assert_equal [['012', false], ['34', true], ['..', false]],
fzf.format('0123456789', 7, [[3, 5]])
# Multi-region
assert_equal [["0", true], ["1", false], ["2", true], ["34..", false]],
fzf.format('0123456789', 7, [[0, 1], [2, 3]])
assert_equal [["..", false], ["5", true], ["6", false], ["78", true], ["9", false]],
fzf.format('0123456789', 7, [[3, 6], [7, 9]])
assert_equal [["..", false], ["3", true], ["4", false], ["5", true], ["..", false]],
fzf.format('0123456789', 7, [[3, 4], [5, 6]])
# Multi-region Overlap
assert_equal [["..", false], ["345", true], ["..", false]],
fzf.format('0123456789', 7, [[4, 5], [3, 6]])
end
def test_fuzzy_matcher
matcher = FZF::FuzzyMatcher.new Regexp::IGNORECASE
list = %w[
juice
juiceful
juiceless
juicily
juiciness
juicy]
assert matcher.cache.empty?
assert_equal(
[["juice", [[0, 1]]],
["juiceful", [[0, 1]]],
["juiceless", [[0, 1]]],
["juicily", [[0, 1]]],
["juiciness", [[0, 1]]],
["juicy", [[0, 1]]]], matcher.match(list, 'j', '', '').sort)
assert !matcher.cache.empty?
assert_equal [list.object_id], matcher.cache.keys
assert_equal 1, matcher.cache[list.object_id].length
assert_equal 6, matcher.cache[list.object_id]['j'].length
assert_equal(
[["juicily", [[0, 5]]],
["juiciness", [[0, 5]]]], matcher.match(list, 'jii', '', '').sort)
assert_equal(
[["juicily", [[2, 5]]],
["juiciness", [[2, 5]]]], matcher.match(list, 'ii', '', '').sort)
assert_equal 3, matcher.cache[list.object_id].length
assert_equal 2, matcher.cache[list.object_id]['ii'].length
# TODO : partial_cache
end
def test_sort_by_rank
matcher = FZF::FuzzyMatcher.new Regexp::IGNORECASE
list = %w[
0____1
0_____1
01
____0_1
01_
_01_
0______1
___01___
]
assert_equal %w[01 01_ _01_ ___01___ ____0_1 0____1 0_____1 0______1],
FZF.new([]).sort_by_rank(matcher.match(list, '01', '', '')).map(&:first)
end
end