diff --git a/.gitignore b/.gitignore index 1377554..b782f12 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,2 @@ *.swp +Gemfile.lock diff --git a/.rspec b/.rspec new file mode 100644 index 0000000..53607ea --- /dev/null +++ b/.rspec @@ -0,0 +1 @@ +--colour diff --git a/Gemfile b/Gemfile new file mode 100644 index 0000000..9da1503 --- /dev/null +++ b/Gemfile @@ -0,0 +1,7 @@ +source 'https://rubygems.org' + +gem 'nokogiri', '1.6.7.2' + +gem 'rspec', '3.4.0' +gem 'rspec-its', '1.2.0' +gem 'pry-nav', '0.2.4' diff --git a/README.md b/README.md index 76b483b..c95474d 100644 --- a/README.md +++ b/README.md @@ -14,9 +14,27 @@ HISTORY INSTALL ------- - * Requires: gem install hpricot - * Install with pathogen: clone/submodule into vim/bundle - * **rbenv** users will need to install hpricot using system ruby `RBENV_VERSION=system sudo gem install hpricot` + +# External dependencies + +Because of ending of `hpricot` gem maintenance was made decision about moving into `nokogiri`. So this gem is required for `vim-rspec` work. +You may install it manually by run `gem install nokogiri` or go into plugin folder and run `bundle`. + +# VIM plugin managers + +## Pathogen +Install with pathogen: clone/submodule into vim/bundle + +## VIM-PLUG +Added into `~/.vimrc` the following line: + +``` +call plug#begin('~/.vim/plugged') + +Plug '' + +call plug#end() +``` USAGE ----- diff --git a/plugin/lib/context_renderer.rb b/plugin/lib/context_renderer.rb deleted file mode 100644 index ae9d2db..0000000 --- a/plugin/lib/context_renderer.rb +++ /dev/null @@ -1,33 +0,0 @@ -# -*- encoding : utf-8 -*- -class RSpecContextRenderer - # context: an html representation of an rspec context from rspec output - # counts: a hash with :passed, :failed, :not_implemented counters - def initialize(context, counts) - @context=context - @counts=counts - @classes = {"passed"=>"+","failed"=>"-","not_implemented"=>"#"} - render_context_header - render_specs - puts " " - end - - private - def render_context_header - puts "[#{(@context/"dl/dt").inner_html}]" - end - - def render_specs - (@context/"dd").each do |dd| - render_spec_descriptor(dd) - FailureRenderer.new(dd/"div[@class~='failure']") if dd[:class] =~ /failed/ - end - end - - def render_spec_descriptor(dd) - txt = (dd/"span:first").inner_html - clazz = dd[:class].gsub(/(?:example|spec) /,'') - puts "#{@classes[clazz]} #{txt}" - outcome = clazz.to_sym - @counts[outcome] += 1 - end -end diff --git a/plugin/lib/failure_renderer.rb b/plugin/lib/failure_renderer.rb index c45a7bd..40647b0 100644 --- a/plugin/lib/failure_renderer.rb +++ b/plugin/lib/failure_renderer.rb @@ -1,9 +1,15 @@ -# -*- encoding : utf-8 -*- +require_relative 'string_util' + class FailureRenderer include StringUtil - def initialize(failure) - @failure = failure + attr_reader :example + + def initialize(example) + @example = example + end + + def render puts failure_message puts failure_location puts backtrace_details @@ -11,23 +17,21 @@ def initialize(failure) private - def indent(msg) - " #{msg}" + def failure_message + indent(unescape(example.failure_message)) end def failure_location unescape( - (@failure/"div[@class='backtrace']/pre").inner_html.split("\n").map { |line| "#{indent(line.strip)}" }.join("\n") + example.failure_location.split("\n") + .map { |line| "#{indent(line.strip)}" } + .join("\n") ) end - def failure_message - indent(unescape((@failure/"div[@class~='message']/pre").inner_html.gsub(/\n/,'').gsub(/\s+/,' '))) - end - def backtrace_details unescape( - backtrace_lines.map do |elem| + example.backtrace_lines.map do |elem| linenum = elem[1] code = elem[3].chomp code = strip_html_spans(code) @@ -35,13 +39,4 @@ def backtrace_details end.join ) end - - def backtrace_lines - (@failure/"pre[@class='ruby']/code").inner_html.scan(/()(\d+)(<\/span>)(.*)/).reject { |line| line[3] =~ ignore_line_if_matches } - end - - def ignore_line_if_matches - /install syntax to get syntax highlighting/ - end - end diff --git a/plugin/lib/rspec_context_renderer.rb b/plugin/lib/rspec_context_renderer.rb new file mode 100644 index 0000000..235387d --- /dev/null +++ b/plugin/lib/rspec_context_renderer.rb @@ -0,0 +1,37 @@ +# +# Renders example group +# +# Parameters: +# example_group - an html representation of an rspec example_group from rspec output +# +class RSpecContextRenderer + attr_reader :example_group + + + def initialize(example_group) + @example_group = example_group + end + + def render + render_example_group_header + render_specs + puts " " + end + + private + + def render_example_group_header + puts "[#{example_group.header_text}]" + end + + def render_specs + example_group.examples.each do |example| + render_spec_descriptor(example) + FailureRenderer.new(example).render if example.failure? + end + end + + def render_spec_descriptor(example) + puts "#{example.status_sign} #{example.header_text}" + end +end diff --git a/plugin/lib/rspec_example_group_result.rb b/plugin/lib/rspec_example_group_result.rb new file mode 100644 index 0000000..a527585 --- /dev/null +++ b/plugin/lib/rspec_example_group_result.rb @@ -0,0 +1,26 @@ +require 'nokogiri' +require_relative 'rspec_example_result' + +class RspecExampleGroupResult + attr_reader :context + + def initialize(context) + @context = context + end + + def header_text + fetch_header_node.inner_html + end + + def examples + context.css('dd').map do |context| + RspecExampleResult.new(context) + end + end + + private + + def fetch_header_node + context.css('dl dt').first + end +end diff --git a/plugin/lib/rspec_example_result.rb b/plugin/lib/rspec_example_result.rb new file mode 100644 index 0000000..da237ff --- /dev/null +++ b/plugin/lib/rspec_example_result.rb @@ -0,0 +1,56 @@ +require 'nokogiri' +require_relative './string_util.rb' + +class RspecExampleResult + include ::StringUtil + + CLASSES_TO_SIGN_MAPPING = {"passed"=>"+","failed"=>"-","not_implemented"=>"#"} + + attr_reader :context + + def initialize(context) + @context = context + end + + def failure? + context['class'] =~ /failed/ + end + + def header_text + context.css('span:first').first.inner_html + end + + def html_class + context['class'].gsub(/(?:example|spec) /,'') + end + + def failure_message + context.css('div.message > pre').first.inner_html + .gsub(/\n/,'').gsub(/\s+/,' ') + end + + def failure_location + unescape( + context.css('div.backtrace > pre').first.inner_html.split("\n").map do |line| + "#{indent(line.strip)}" + end.join("\n") + ) + context.css('div.backtrace > pre').first.inner_html + end + + def backtrace_lines + context.css('pre.ruby > code').first.inner_html + .scan(/()(\d+)(<\/span>)(.*)/) + .reject { |line| line[3] =~ ignore_line_if_matches } + end + + def status_sign + CLASSES_TO_SIGN_MAPPING[html_class] + end + + private + + def ignore_line_if_matches + /install syntax to get syntax highlighting/ + end +end diff --git a/plugin/lib/rspec_test_result.rb b/plugin/lib/rspec_test_result.rb new file mode 100644 index 0000000..5e94772 --- /dev/null +++ b/plugin/lib/rspec_test_result.rb @@ -0,0 +1,54 @@ +require 'nokogiri' +require_relative 'rspec_example_group_result' + +class RspecTestResult + attr_reader :html, :doc + + def initialize(html) + @html = html + @doc = Nokogiri::HTML(html) + end + + def duration_str + fetch_duration_node.inner_html.scan(/".*"/).first.gsub(/<\/?strong>/,"").gsub(/\"/,'') + end + + def examples_count + fetch_counts_str.match(/(\d+) example/)[1].to_i rescue 0 + end + + def failures_count + fetch_counts_str.match(/(\d+) failure/)[1].to_i rescue 0 + end + + def pending_count + fetch_counts_str.match(/(\d+) pending/)[1].to_i rescue 0 + end + + def example_groups + doc.css('div.example_group').map do |context| + RspecExampleGroupResult.new(context) + end + end + + private + + def fetch_counts_str + fetch_counts_node + .inner_html.scan(/".*"/).first + end + + def fetch_counts_node + fetch_script_node_with_inner_pattern(/totals/) + end + + def fetch_duration_node + fetch_script_node_with_inner_pattern(/duration/) + end + + def fetch_script_node_with_inner_pattern(pattern) + doc.css("script") + .select { |script| script.inner_html =~ pattern } + .first + end +end diff --git a/plugin/lib/string_util.rb b/plugin/lib/string_util.rb index dc611a5..b762e06 100644 --- a/plugin/lib/string_util.rb +++ b/plugin/lib/string_util.rb @@ -6,4 +6,8 @@ def strip_html_spans(code) def unescape(html) CGI.unescapeHTML(html) end + + def indent(msg) + " #{msg}" + end end diff --git a/plugin/vim-rspec.rb b/plugin/vim-rspec.rb index d2feff9..d6d806d 100644 --- a/plugin/vim-rspec.rb +++ b/plugin/vim-rspec.rb @@ -1,20 +1,26 @@ # -*- encoding : utf-8 -*- -require "rubygems" -require "hpricot" +require "nokogiri" require 'cgi' -require "#{File.join(File.dirname(__FILE__), "lib/string_util")}" -require "#{File.join(File.dirname(__FILE__), "lib/failure_renderer")}" -require "#{File.join(File.dirname(__FILE__), "lib/context_renderer")}" +require File.join(File.dirname(__FILE__), "lib/string_util") +require File.join(File.dirname(__FILE__), "lib/failure_renderer") +require File.join(File.dirname(__FILE__), "lib/rspec_context_renderer") +require File.join(File.dirname(__FILE__), "lib/rspec_example_result") +require File.join(File.dirname(__FILE__), "lib/rspec_example_group_result") +require File.join(File.dirname(__FILE__), "lib/rspec_test_result") class RSpecOutputHandler - - def initialize(doc) - @doc=doc - @counts={ - :passed => 0, - :failed => 0, - :not_implemented => 0 + attr_reader :test_result + + def initialize(test_result) + @test_result = test_result + @counts = { + passed: 0, + failed: 0, + not_implemented: 0 } + end + + def render render_header render_examples end @@ -22,23 +28,15 @@ def initialize(doc) private def render_header - stats = (@doc/"script").select {|script| script.innerHTML =~ /duration|totals/ } - stats.map! do |script| - script.inner_html.scan(/".*"/).first.gsub(/<\/?strong>/,"").gsub(/\"/,'') - end - # results in ["Finished in 0.00482 seconds", "2 examples, 1 failure"] - failure_success_messages,other_stats = stats.partition {|stat| stat =~ /failure/} - render_red_green_header(failure_success_messages.first) - other_stats.each do |stat| - puts stat - end + render_red_green_header + puts test_result.duration_str puts " " end - def render_red_green_header(failure_success_messages) - total_count = failure_success_messages.match(/(\d+) example/)[1].to_i rescue 0 - fail_count = failure_success_messages.match(/(\d+) failure/)[1].to_i rescue 0 - pending_count = failure_success_messages.match(/(\d+) pending/)[1].to_i rescue 0 + def render_red_green_header + total_count = test_result.examples_count + fail_count = test_result.failures_count + pending_count = test_result.pending_count if fail_count > 0 puts "------------------------------" @@ -49,15 +47,15 @@ def render_red_green_header(failure_success_messages) puts "+ PASS: All #{total_count} Specs Pass!" puts "++++++++++++++++++++++++++++++" end - end def render_examples - (@doc/"div[@class~='example_group']").each do |context| - RSpecContextRenderer.new(context, @counts) + test_result.example_groups.each do |example_group| + RSpecContextRenderer.new(example_group).render end end - end -renderer = RSpecOutputHandler.new(Hpricot(STDIN.read)) +test_result = RspecTestResult.new(STDIN.read) + +RSpecOutputHandler.new(test_result).render diff --git a/plugin/vim-rspec.vim b/plugin/vim-rspec.vim index 4a5b950..73527ad 100644 --- a/plugin/vim-rspec.vim +++ b/plugin/vim-rspec.vim @@ -15,119 +15,50 @@ " * g:RspecOpts :: Opts to send to rspec call " * g:RspecSplitHorizontal :: Set to 0 to cause Vertical split (default:1) -let s:hpricot_cmd = "" -let s:hpricot = 0 -let s:helper_dir = expand(":h") +let s:plugin_dir = expand(":h") + if !exists("g:RspecKeymap") let g:RspecKeymap=1 end -function! s:find_hpricot() - return system("( gem search -i hpricot &> /dev/null && echo true ) - \ || ( dpkg-query -Wf'${db:Status-abbrev}' ruby-hpricot 2>/dev/null - \ | grep -q '^i' && echo true ) - \ || echo false") -endfunction - -function! s:error_msg(msg) - echohl ErrorMsg - echo a:msg - echohl None -endfunction - -function! s:notice_msg(msg) - echohl MoreMsg - echo a:msg - echohl None -endfunction - -function! s:fetch(varname, default) - if exists("g:".a:varname) - return eval("g:".a:varname) - else - return a:default - endif -endfunction - -function! s:createOutputWin() - if !exists("g:RspecSplitHorizontal") - let g:RspecSplitHorizontal=1 - endif - - let splitLocation = "botright " - let splitSize = 15 - - if bufexists('RSpecOutput') - silent! bw! RSpecOutput - end - - if g:RspecSplitHorizontal == 1 - silent! exec splitLocation . ' ' . splitSize . ' new' - else - silent! exec splitLocation . ' ' . splitSize . ' vnew' - end - silent! exec "edit RSpecOutput" -endfunction +execute "source " . s:plugin_dir . "/vim_rspec/helpers.vim" +execute "source " . s:plugin_dir . "/vim_rspec/win_cmd.vim" function! s:RunSpecMain(type) let l:bufn = expand("%:p") let s:SpecFile = l:bufn - if len(s:hpricot_cmd)<1 - let s:hpricot_cmd = s:find_hpricot() - let s:hpricot = match(s:hpricot_cmd,'true')>=0 - end - - if !s:hpricot - call s:error_msg("You need the hpricot gem to run this script.") - return - end + call vim_rspec#helpers#check_nokogiri() " find the installed rspec command - let l:default_cmd = "" - if executable("spec")==1 - let l:default_cmd = "spec" - elseif executable("rspec")==1 - let l:default_cmd = "rspec" - end - - " filter - let l:filter = "ruby ". s:fetch("RspecRBPath", s:helper_dir."/vim-rspec.rb") + let l:default_cmd = vim_rspec#helpers#find_rspec_executable() " run just the current file - if a:type=="file" - if match(l:bufn,'_spec.rb')>=0 - call s:notice_msg("Running " . s:SpecFile . "...") - let l:spec_bin = s:fetch("RspecBin",l:default_cmd) - let l:spec_opts = s:fetch("RspecOpts", "") - let l:spec = l:spec_bin . " " . l:spec_opts . " -f h " . l:bufn - else - call s:error_msg("Seems ".l:bufn." is not a *_spec.rb file") + if a:type == "file" + try + let l:spec = vim_rspec#runners#file() + catch /^Seems/ + call vim_rspec#helpers#error_msg(v:exception) return - end - elseif a:type=="line" - if match(l:bufn,'_spec.rb')>=0 - let l:current_line = line('.') - - call s:notice_msg("Running Line " . l:current_line . " on " . s:SpecFile . " ") - let l:spec_bin = s:fetch("RspecBin",l:default_cmd) - let l:spec_opts = s:fetch("RspecOpts", "") - let l:spec = l:spec_bin . " " . l:spec_opts . " -l " . l:current_line . " -f h " . l:bufn - else - call s:error_msg("Seems ".l:bufn." is not a *_spec.rb file") + endtry + elseif a:type == "line" + try + let l:spec = vim_rspec#runners#line() + catch /^Seems/ + call vim_rspec#helpers#error_msg(v:exception) return - end - elseif a:type=="rerun" + endtry + elseif a:type == "rerun" if exists("s:spec") let l:spec = s:spec else - call s:error_msg("No rspec run to repeat") + call vim_rspec#helpers#error_msg("No rspec run to repeat") return end else let l:dir = expand("%:p:h") if isdirectory(l:dir."/spec")>0 - call s:notice_msg("Running spec on the spec directory ...") + call vim_rspec#helpers#notice_msg("Running spec on the spec directory ...") else " try to find a spec directory on the current path let l:tokens = split(l:dir,"/") @@ -141,74 +72,40 @@ function! s:RunSpecMain(type) end endfor if len(l:dir)>0 - call s:notice_msg("Running spec with on the spec directory found (".l:dir.") ...") + call vim_rspec#helpers#notice_msg("Running spec with on the spec directory found (".l:dir.") ...") else - call s:error_msg("No ".l:dir."/spec directory found") + call vim_rspec#helpers#error_msg("No ".l:dir."/spec directory found") return end end if isdirectory(l:dir)<0 - call s:error_msg("Could not find the ".l:dir." directory.") + call vim_rspec#helpers#error_msg("Could not find the ".l:dir." directory.") return end - let l:spec = s:fetch("RspecBin", l:default_cmd) . s:fetch("RspecOpts", "") + let l:spec = vim_rspec#helpers#fetch_var("RspecBin", l:default_cmd) . vim_rspec#helpers#fetch_var("RspecOpts", "") let l:spec = l:spec . " -f h " . l:dir . " -p **/*_spec.rb" end let s:spec = l:spec " run the spec command + let l:filter = vim_rspec#helpers#build_filter_command(s:plugin_dir) + let s:cmd = l:spec." | ".l:filter "put the result on a new buffer - call s:createOutputWin() + call vim_rspec#win_cmd#create_output_win() setl buftype=nofile silent exec "r! ".s:cmd setl syntax=vim-rspec - silent exec "nnoremap :call TryToOpen()" - silent exec 'nnoremap n /\/.*spec.*\::call TryToOpen()' - silent exec 'nnoremap N ?/\/.*spec.*\::call TryToOpen()' + silent exec "nnoremap :call vim_rspec#win_cmd#try_to_open('" . s:SpecFile . "')" + silent exec "nnoremap n /\/.*spec.*\::call vim_rspec#win_cmd#try_to_open('" . s:SpecFile . "')" + silent exec "nnoremap N ?/\/.*spec.*\::call vim_rspec#win_cmd#try_to_open('" . s:SpecFile . "')" silent exec "nnoremap q :q:wincmd p" setl nolist setl foldmethod=expr setl foldexpr=getline(v:lnum)=~'^\+' setl foldtext=\"+--\ \".string(v:foldend-v:foldstart+1).\"\ passed\ \" call cursor(1,1) - -endfunction - -function! s:FindWindowByBufferName(buffername) - let l:windowNumberToBufnameList = map(range(1, winnr('$')), '[v:val, bufname(winbufnr(v:val))]') - let l:arrayIndex = match(l:windowNumberToBufnameList, a:buffername) - let l:windowNumber = windowNumberToBufnameList[l:arrayIndex][0] - return l:windowNumber -endfunction - -function! s:SwitchToWindowNumber(number) - exe a:number . "wincmd w" -endfunction - -function! s:SwitchToWindowByName(buffername) - let l:windowNumber = s:FindWindowByBufferName(a:buffername) - call s:SwitchToWindowNumber(l:windowNumber) -endfunction - -function! s:TryToOpen() - " Search up to find '*_spec.rb' - call search("_spec","bcW") - let l:line = getline(".") - if match(l:line,'^ .*_spec.rb')<0 - call s:error_msg("No spec file found.") - return - end - let l:tokens = split(l:line,":") - - " move back to the other window, if available - call s:SwitchToWindowByName(s:SpecFile) - - " open the file in question (either in the split) - " that was already open, or in the current win - exec "e ".substitute(l:tokens[0],'/^\s\+',"","") - call cursor(l:tokens[1],1) endfunction function! RunSpec() diff --git a/plugin/vim_rspec/helpers.vim b/plugin/vim_rspec/helpers.vim new file mode 100644 index 0000000..aca1b03 --- /dev/null +++ b/plugin/vim_rspec/helpers.vim @@ -0,0 +1,49 @@ +function! vim_rspec#helpers#check_nokogiri() + let s:nokogiri = s:find_nokogiri() + let s:nokogiri = match(s:nokogiri,'true') >= 0 + if !s:nokogiri + throw("You need the `nokogiri` gem to run this script.") + end +endfunction + +function! vim_rspec#helpers#error_msg(msg) + echohl ErrorMsg + echo a:msg + echohl None +endfunction + +function! vim_rspec#helpers#notice_msg(msg) + echohl MoreMsg + echo a:msg + echohl None +endfunction + +function! vim_rspec#helpers#fetch_var(varname, default) + if exists("g:".a:varname) + return eval("g:".a:varname) + else + return a:default + endif +endfunction + +function! vim_rspec#helpers#find_rspec_executable() + if executable("spec") == 1 + return "spec" + elseif executable("rspec") == 1 + return "rspec" + end + return "" +endfunction + +function! vim_rspec#helpers#build_filter_command(plugin_dir) + let default_filter_cmd = a:plugin_dir . "/vim-rspec.rb" + let rb_path = vim_rspec#helpers#fetch_var("RspecRBPath", default_filter_cmd) + return "ruby " . rb_path +endfunction + +"====== +" Local functions +"====== +function! s:find_nokogiri() + return system("gem search -i nokogiri &> /dev/null && echo true") +endfunction diff --git a/plugin/vim_rspec/runners.vim b/plugin/vim_rspec/runners.vim new file mode 100644 index 0000000..b606339 --- /dev/null +++ b/plugin/vim_rspec/runners.vim @@ -0,0 +1,36 @@ +function! vim_rspec#runners#file() + call s:check_is_buffer_spec_file() + + call vim_rspec#helpers#notice_msg("Running " . s:buffer_name() . "...") + return s:fetch_spec_bin() . " " . s:fetch_spec_opts() . " -f h " . s:buffer_name() +endfunction + +function! vim_rspec#runners#line() + call s:check_is_buffer_spec_file() + + let l:current_line = line('.') + call vim_rspec#helpers#notice_msg("Running Line " . l:current_line . " on " . s:buffer_name() . " ") + return s:fetch_spec_bin() . " " . s:fetch_spec_opts() . " -f h " . s:buffer_name() . ":" . l:current_line +endfunction + +"====== +" Local functions +"====== +function! s:check_is_buffer_spec_file() + let l:bufn = s:buffer_name() + if match(l:bufn,'_spec.rb') <= 0 + throw("Seems ".l:bufn." is not a *_spec.rb file") + endif +endfunction + +function! s:buffer_name() + return expand("%:p") +endfunction + +function! s:fetch_spec_bin() + return vim_rspec#helpers#fetch_var("RspecBin", vim_rspec#helpers#find_rspec_executable()) +endfunction + +function! s:fetch_spec_opts() + return vim_rspec#helpers#fetch_var("RspecOpts", "") +endfunction diff --git a/plugin/vim_rspec/win_cmd.vim b/plugin/vim_rspec/win_cmd.vim new file mode 100644 index 0000000..ec47cca --- /dev/null +++ b/plugin/vim_rspec/win_cmd.vim @@ -0,0 +1,54 @@ +function! vim_rspec#win_cmd#create_output_win() + let isSplitHorizontal = vim_rspec#helpers#fetch_var("RspecSplitHorizontal", 1) + let splitDirCommand = isSplitHorizontal == 1 ? 'new' : 'vnew' + + let splitLocation = "botright " + let splitSize = 15 + + if bufexists('RSpecOutput') + silent! bw! RSpecOutput + end + + silent! exec splitLocation . ' ' . splitSize . ' ' . splitDirCommand + silent! exec "edit RSpecOutput" +endfunction + +function! vim_rspec#win_cmd#try_to_open(spec_file) + " Search up to find '*_spec.rb' + call search("_spec","bcW") + let l:line = getline(".") + if match(l:line,'^ .*_spec.rb')<0 + call vim_rspec#helpers#error_msg("No spec file found.") + return + end + let l:tokens = split(l:line,":") + + " move back to the other window, if available + call s:SwitchToWindowByName(a:spec_file) + + " open the file in question (either in the split) + " that was already open, or in the current win + exec "e ".substitute(l:tokens[0],'/^\s\+',"","") + call cursor(l:tokens[1],1) +endfunction + +"====== +" Local functions +"====== +function! s:SwitchToWindowByName(buffername) + let l:windowNumber = s:FindWindowByBufferName(a:buffername) + call s:SwitchToWindowNumber(l:windowNumber) +endfunction + +function! s:FindWindowByBufferName(buffername) + let l:windowNumberToBufnameList = map(range(1, winnr('$')), '[v:val, bufname(winbufnr(v:val))]') + let l:arrayIndex = match(l:windowNumberToBufnameList, a:buffername) + let l:windowNumber = windowNumberToBufnameList[l:arrayIndex][0] + return l:windowNumber +endfunction + +function! s:SwitchToWindowNumber(number) + exe a:number . "wincmd w" +endfunction + + diff --git a/spec/plugin/lib/rspec_example_group_result_spec.rb b/spec/plugin/lib/rspec_example_group_result_spec.rb new file mode 100644 index 0000000..a98a553 --- /dev/null +++ b/spec/plugin/lib/rspec_example_group_result_spec.rb @@ -0,0 +1,25 @@ +require 'spec_helper' +require './plugin/lib/rspec_example_group_result' + +describe RspecExampleGroupResult do + subject(:result) { described_class.new(doc) } + + let(:doc) { Nokogiri::HTML(input_html) } + let(:input_html) { File.read(RspecFileExampleProvider.example_group_path) } + + describe '#header_text' do + it 'returns proper string' do + expect(result.header_text).to eq 'EmptyUser' + end + end + + describe '#examples' do + it 'returns proper number of examples' do + expect(result.examples.count).to eq 7 + end + + it 'returns wraps examples' do + expect(result.examples.first).to be_kind_of(RspecExampleResult) + end + end +end diff --git a/spec/plugin/lib/rspec_example_result_spec.rb b/spec/plugin/lib/rspec_example_result_spec.rb new file mode 100644 index 0000000..4b32d4c --- /dev/null +++ b/spec/plugin/lib/rspec_example_result_spec.rb @@ -0,0 +1,71 @@ +require 'spec_helper' +require './plugin/lib/rspec_example_result' + +describe RspecExampleResult do + subject(:result) { described_class.new(doc) } + + let(:doc) { Nokogiri::HTML.fragment(input_html).css('dd').first } + let(:input_html) { File.read(RspecFileExampleProvider.passed_example_path) } + + describe '#header_text' do + it 'returns proper text' do + expect(result.header_text).to eq 'should respond to #member?' + end + end + + context 'when passed example' do + describe '#failure?' do + its(:failure?) { is_expected.to be_falsey } + end + + describe '#html_class' do + it 'returns proper value' do + expect(result.html_class).to eq 'passed' + end + end + + describe '#status_sign' do + it 'returns proper sign' do + expect(result.status_sign).to eq '+' + end + end + end + + context 'when failure example' do + let(:input_html) { File.read(RspecFileExampleProvider.failure_example_path) } + + describe '#failure?' do + its(:failure?) { is_expected.to be_truthy } + end + + describe '#html_class' do + it 'returns proper value' do + expect(result.html_class).to eq 'failed' + end + end + + describe '#failure_message' do + it 'returns proper value' do + expect(result.failure_message).to eq 'expected #<EmptyUser:0x007f95646fdaf0 @role="guest", @id=0> to respond to :forget_me2!' + end + end + + describe '#failure_location' do + it 'returns proper value' do + expect(result.failure_location).to eq "./spec/app/core/empty_user_spec.rb:18:in `block (2 levels) in <top (required)>'" + end + end + + describe '#backtrace_lines' do + it 'returns proper values' do + expect(result.backtrace_lines).to eq [["", "16", "", " end"], ["", "17", "", ""], ["", "18", "", " it { is_expected.to respond_to :forget_me2! }"], ["", "19", "", " it { is_expected.to respond_to :force_forget_me! }"], ["", "20", "", "end"]] + end + end + + describe '#status_sign' do + it 'returns proper sign' do + expect(result.status_sign).to eq '-' + end + end + end +end diff --git a/spec/plugin/lib/rspec_test_result_spec.rb b/spec/plugin/lib/rspec_test_result_spec.rb new file mode 100644 index 0000000..b963e64 --- /dev/null +++ b/spec/plugin/lib/rspec_test_result_spec.rb @@ -0,0 +1,42 @@ +require 'spec_helper' +require './plugin/lib/rspec_test_result' + +describe RspecTestResult do + subject(:result) { described_class.new(input_html) } + + let(:input_html) { File.read(RspecFileExampleProvider.not_all_passed) } + + describe '#duration_str' do + it 'returns proper duration string' do + expect(result.duration_str).to eq 'Finished in 0.80876 seconds' + end + end + + describe '#examples_count' do + it 'returns proper overall examples count' do + expect(result.examples_count).to eq 15 + end + end + + describe '#failures_count' do + it 'returns proper failure examples count' do + expect(result.failures_count).to eq 1 + end + end + + describe '#pending_count' do + it 'returns proper pending examples count' do + expect(result.pending_count).to eq 0 + end + end + + describe '#example_groups' do + it 'returns proper number of example groups' do + expect(result.example_groups.count).to eq 12 + end + + it 'returns wrapped example groups' do + expect(result.example_groups.first).to be_kind_of(RspecExampleGroupResult) + end + end +end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb new file mode 100644 index 0000000..3ef4126 --- /dev/null +++ b/spec/spec_helper.rb @@ -0,0 +1,33 @@ +$TESTING=true + +require 'rspec/its' +require 'pry-nav' + +Dir[File.join(File.dirname(__FILE__), 'support/**/*.rb')].each { |f| require f } + +class RspecFileExampleProvider + def self.not_all_passed + File.join File.dirname(__FILE__), 'support/not_all_passed.html' + end + + def self.example_group_path + File.join File.dirname(__FILE__), 'support/example_group.html' + end + + def self.passed_example_path + File.join File.dirname(__FILE__), 'support/passed_example.html' + end + + def self.failure_example_path + File.join File.dirname(__FILE__), 'support/failure_example.html' + end +end + +RSpec.configure do |config| + config.mock_with :rspec + + config.filter_run focus: true + config.run_all_when_everything_filtered = true + + config.order = "random" +end diff --git a/spec/support/example_group.html b/spec/support/example_group.html new file mode 100644 index 0000000..cb61246 --- /dev/null +++ b/spec/support/example_group.html @@ -0,0 +1,34 @@ +
+
+
EmptyUser
+ +
should respond to #guest?0.00322s
+ + + + +
+ should respond to #forget_me2! + 0.02465s +
+
expected #<EmptyUser:0x007f95646fdaf0 @role="guest", @id=0> to respond to :forget_me2!
+
./spec/app/core/empty_user_spec.rb:18:in `block (2 levels) in <top (required)>'
+
16  end
+17
+18  it { is_expected.to respond_to :forget_me2! }
+19  it { is_expected.to respond_to :force_forget_me! }
+20end
+
+
+ +
should respond to #admin?0.00176s
+ +
should respond to #force_forget_me!0.00126s
+ +
should respond to #superadmin?0.00104s
+ +
should respond to #moderator?0.00098s
+ +
should respond to #member?0.00095s
+
+
diff --git a/spec/support/failure_example.html b/spec/support/failure_example.html new file mode 100644 index 0000000..4bccf13 --- /dev/null +++ b/spec/support/failure_example.html @@ -0,0 +1,13 @@ +
+should respond to #forget_me2! +0.02465s +
+
expected #<EmptyUser:0x007f95646fdaf0 @role="guest", @id=0> to respond to :forget_me2!
+
./spec/app/core/empty_user_spec.rb:18:in `block (2 levels) in <top (required)>'
+
16  end
+17
+18  it { is_expected.to respond_to :forget_me2! }
+19  it { is_expected.to respond_to :force_forget_me! }
+20end
+
+
diff --git a/spec/support/not_all_passed.html b/spec/support/not_all_passed.html new file mode 100644 index 0000000..49cf413 --- /dev/null +++ b/spec/support/not_all_passed.html @@ -0,0 +1,395 @@ +Run options: include {:focus=>true} + +All examples were filtered out; ignoring {:focus=>true} + + + + RSpec results + + + + + + + + +
+ +
+
+

RSpec Code Examples

+
+ +
+ + + +
+ +
+

 

+

 

+
+
+ + +
+
+
+
EmptyUser
+ +
should respond to #guest?0.00322s
+ + + + +
+ should respond to #forget_me2! + 0.02465s +
+
expected #<EmptyUser:0x007f95646fdaf0 @role="guest", @id=0> to respond to :forget_me2!
+
./spec/app/core/empty_user_spec.rb:18:in `block (2 levels) in <top (required)>'
+
16  end
+17
+18  it { is_expected.to respond_to :forget_me2! }
+19  it { is_expected.to respond_to :force_forget_me! }
+20end
+
+
+ +
should respond to #admin?0.00176s
+ +
should respond to #force_forget_me!0.00126s
+ +
should respond to #superadmin?0.00104s
+ +
should respond to #moderator?0.00098s
+ +
should respond to #member?0.00095s
+
+
+
+
+
when guest user
+
+
+
+
+
member?
+ +
should be falsey0.00112s
+
+
+
+
+
registered?
+ +
should be falsey0.00129s
+
+
+
+
+
guest?
+ +
should be truthy0.00156s
+
+
+
+
+
when registered user
+
+
+
+
+
guest?
+ +
should be falsey0.18902s
+
+
+
+
+
registered?
+ +
should be truthy0.00971s
+
+
+
+
+
member?
+ +
should be truthy0.01008s
+
+
+
+
+
#posted?
+ +
reutnrs false0.00168s
+
+
+
+
+
#new_record?
+
+
+
+
+
new_record?
+ +
should be truthy0.00149s
+
+
+ + +
+
+ + diff --git a/spec/support/passed_example.html b/spec/support/passed_example.html new file mode 100644 index 0000000..17bd25c --- /dev/null +++ b/spec/support/passed_example.html @@ -0,0 +1 @@ +
should respond to #member?0.00095s