# Copyright (C) 2015-2018 YouCompleteMe contributors # # This file is part of YouCompleteMe. # # YouCompleteMe is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # YouCompleteMe is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with YouCompleteMe. If not, see . from ycm.tests import PathToTestFile from ycm.tests.test_utils import ( CurrentWorkingDirectory, ExtendedMock, MockVimBuffers, MockVimModule, Version, VimBuffer, VimError, WindowsAndMacOnly ) MockVimModule() from ycm import vimsupport from hamcrest import ( assert_that, calling, contains_exactly, empty, equal_to, has_entry, raises ) from unittest import TestCase from unittest.mock import MagicMock, call, patch from ycmd.utils import ToBytes import os import json def AssertBuffersAreEqualAsBytes( result_buffer, expected_buffer ): assert_that( len( result_buffer ), equal_to( len( expected_buffer ) ) ) for result_line, expected_line in zip( result_buffer, expected_buffer ): assert_that( ToBytes( result_line ), equal_to( ToBytes( expected_line ) ) ) def _BuildChunk( start_line, start_column, end_line, end_column, replacement_text, filepath='test_file_name' ): return { 'range': { 'start': { 'filepath': filepath, 'line_num': start_line, 'column_num': start_column, }, 'end': { 'filepath': filepath, 'line_num': end_line, 'column_num': end_column, }, }, 'replacement_text': replacement_text } def _BuildLocations( start_line, start_column, end_line, end_column ): return { 'line_num' : start_line, 'column_num': start_column, }, { 'line_num' : end_line, 'column_num': end_column, } class VimsupportTest( TestCase ): @patch( 'ycm.vimsupport.WinIDForWindow', side_effect = range( 1001, 1010 ) ) @patch( 'vim.eval', new_callable = ExtendedMock ) def test_SetLocationListsForBuffer_Current( self, vim_eval, *args ): diagnostics = [ { 'bufnr': 3, 'filename': 'some_filename', 'lnum': 5, 'col': 22, 'type': 'E', 'valid': 1 } ] current_buffer = VimBuffer( '/test', number = 3 ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): vimsupport.SetLocationListsForBuffer( 3, diagnostics ) vim_eval.assert_has_exact_calls( [ call( 'setloclist( 1001, [], " ", { "title": "ycm_loc", ' '"items": [{"bufnr": 3, "filename": "some_filename", "lnum": 5, ' '"col": 22, "type": "E", "valid": 1}] } )' ), call( 'getloclist( 1001, { "nr": "$", "id": 0 } ).id' ), ] ) @patch( 'vim.eval', new_callable = ExtendedMock ) def test_SetLocationListsForBuffer_NotCurrent( self, vim_eval ): diagnostics = [ { 'bufnr': 3, 'filename': 'some_filename', 'lnum': 5, 'col': 22, 'type': 'E', 'valid': 1 } ] current_buffer = VimBuffer( '/test', number = 3 ) other_buffer = VimBuffer( '/notcurrent', number = 1 ) with MockVimBuffers( [ current_buffer, other_buffer ], [ current_buffer ] ): vimsupport.SetLocationListsForBuffer( 1, diagnostics ) vim_eval.assert_not_called() @patch( 'vim.eval', new_callable = ExtendedMock, side_effect = [ -1, 1 ] ) def test_SetLocationListsForBuffer_NotVisible( self, vim_eval ): diagnostics = [ { 'bufnr': 3, 'filename': 'some_filename', 'lnum': 5, 'col': 22, 'type': 'E', 'valid': 1 } ] current_buffer = VimBuffer( '/test', number = 3 ) other_buffer = VimBuffer( '/notcurrent', number = 1 ) with MockVimBuffers( [ current_buffer, other_buffer ], [ current_buffer ] ): vimsupport.SetLocationListsForBuffer( 1, diagnostics ) vim_eval.assert_not_called() @patch( 'ycm.vimsupport.WinIDForWindow', side_effect = range( 1001, 1010 ) ) @patch( 'vim.eval', new_callable = ExtendedMock, side_effect = [ -1, 1 ] ) def test_SetLocationListsForBuffer_MultipleWindows( self, vim_eval, *args ): diagnostics = [ { 'bufnr': 3, 'filename': 'some_filename', 'lnum': 5, 'col': 22, 'type': 'E', 'valid': 1 } ] current_buffer = VimBuffer( '/test', number = 3 ) other_buffer = VimBuffer( '/notcurrent', number = 1 ) with MockVimBuffers( [ current_buffer, other_buffer ], [ current_buffer, other_buffer ] ): vimsupport.SetLocationListsForBuffer( 1, diagnostics ) vim_eval.assert_has_exact_calls( [ call( 'setloclist( 1001, [], " ", { "title": "ycm_loc", "items": ' f'{ json.dumps( diagnostics ) } }} )' ), call( 'getloclist( 1001, { "nr": "$", "id": 0 } ).id' ), ] ) @patch( 'ycm.vimsupport.WinIDForWindow', side_effect = range( 1001, 1010 ) ) @patch( 'vim.eval', new_callable = ExtendedMock ) def test_SetLocationList( self, vim_eval, *args ): diagnostics = [ { 'bufnr': 3, 'filename': 'some_filename', 'lnum': 5, 'col': 22, 'type': 'E', 'valid': 1 } ] current_buffer = VimBuffer( '/test', number = 3 ) with MockVimBuffers( [ current_buffer ], [ current_buffer ], ( 1, 1 ) ): vimsupport.SetLocationList( diagnostics ) vim_eval.assert_has_exact_calls( [ call( 'setloclist( 1001, [], " ", { "title": "ycm_loc", "items": ' '[{"bufnr": 3, "filename": "some_filename", "lnum": ' '5, "col": 22, "type": "E", "valid": 1}] } )' ), call( 'getloclist( 1001, { "nr": "$", "id": 0 } ).id' ), ] ) @patch( 'ycm.vimsupport.WinIDForWindow', side_effect = range( 1001, 1010 ) ) @patch( 'vim.eval', new_callable = ExtendedMock ) def test_SetLocationList_NotCurrent( self, vim_eval, *args ): diagnostics = [ { 'bufnr': 3, 'filename': 'some_filename', 'lnum': 5, 'col': 22, 'type': 'E', 'valid': 1 } ] current_buffer = VimBuffer( '/test', number = 3 ) other_buffer = VimBuffer( '/notcurrent', number = 1 ) with MockVimBuffers( [ current_buffer, other_buffer ], [ current_buffer, other_buffer ], ( 1, 1 ) ): vimsupport.SetLocationList( diagnostics ) # This version does not check the current # buffer and just sets the current win vim_eval.assert_has_exact_calls( [ call( 'setloclist( 1001, [], " ", { "title": "ycm_loc", "items": ' '[{"bufnr": 3, "filename": "some_filename", "lnum": 5, "col": 22, ' '"type": "E", "valid": 1}] } )' ), call( 'getloclist( 1001, { "nr": "$", "id": 0 } ).id' ), ] ) @patch( 'ycm.vimsupport.VariableExists', return_value = True ) @patch( 'ycm.vimsupport.SetFittingHeightForCurrentWindow' ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_OpenLocationList( self, vim_command, fitting_height, variable_exists ): vimsupport.OpenLocationList( focus = False, autoclose = True ) vim_command.assert_has_exact_calls( [ call( 'lopen' ), call( 'augroup ycmlocation' ), call( 'autocmd! * ' ), call( 'autocmd WinLeave ' 'if bufnr( "%" ) == expand( "" ) | q | endif ' '| autocmd! ycmlocation' ), call( 'augroup END' ), call( 'doautocmd User YcmLocationOpened' ), call( 'silent! wincmd p' ) ] ) fitting_height.assert_called_once_with() variable_exists.assert_called_once_with( '#User#YcmLocationOpened' ) @patch( 'vim.command' ) def test_SetFittingHeightForCurrentWindow_LineWrapOn( self, vim_command, *args ): # Create a two lines buffer whose first # line is longer than the window width. current_buffer = VimBuffer( 'buffer', contents = [ 'a' * 140, 'b' * 80 ] ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.width = 120 vim.current.window.options[ 'wrap' ] = True vimsupport.SetFittingHeightForCurrentWindow() vim_command.assert_called_once_with( '3wincmd _' ) @patch( 'vim.command' ) def test_SetFittingHeightForCurrentWindow_NoResize( self, vim_command, *args ): # Create a two lines buffer whose first # line is longer than the window width. current_buffer = VimBuffer( 'buffer', contents = [ 'a' * 140, 'b' * 80 ], vars = { 'ycm_no_resize': 1 } ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.width = 120 vim.current.window.options[ 'wrap' ] = True vimsupport.SetFittingHeightForCurrentWindow() vim_command.assert_not_called() @patch( 'vim.command' ) def test_SetFittingHeightForCurrentWindow_LineWrapOff( self, vim_command, *args ): # Create a two lines buffer whose first # line is longer than the window width. current_buffer = VimBuffer( 'buffer', contents = [ 'a' * 140, 'b' * 80 ] ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.width = 120 vim.current.window.options[ 'wrap' ] = False vimsupport.SetFittingHeightForCurrentWindow() vim_command.assert_called_once_with( '2wincmd _' ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Repl_1( self ): # Replace with longer range result_buffer = VimBuffer( 'buffer', contents = [ 'This is a string' ] ) start, end = _BuildLocations( 1, 11, 1, 17 ) vimsupport.ReplaceChunk( start, end, 'pie', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a pie' ], result_buffer ) # and replace again start, end = _BuildLocations( 1, 10, 1, 11 ) vimsupport.ReplaceChunk( start, end, ' piece of ', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a piece of pie' ], result_buffer ) # and once more, for luck start, end = _BuildLocations( 1, 1, 1, 5 ) vimsupport.ReplaceChunk( start, end, 'How long', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'How long is a piece of pie' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Repl_2( self ): # Replace with shorter range result_buffer = VimBuffer( 'buffer', contents = [ 'This is a string' ] ) start, end = _BuildLocations( 1, 11, 1, 17 ) vimsupport.ReplaceChunk( start, end, 'test', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a test' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Repl_3( self ): # Replace with equal range result_buffer = VimBuffer( 'buffer', contents = [ 'This is a string' ] ) start, end = _BuildLocations( 1, 6, 1, 8 ) vimsupport.ReplaceChunk( start, end, 'be', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This be a string' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Add_1( self ): # Insert at start result_buffer = VimBuffer( 'buffer', contents = [ 'is a string' ] ) start, end = _BuildLocations( 1, 1, 1, 1 ) vimsupport.ReplaceChunk( start, end, 'This ', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a string' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Add_2( self ): # Insert at end result_buffer = VimBuffer( 'buffer', contents = [ 'This is a ' ] ) start, end = _BuildLocations( 1, 11, 1, 11 ) vimsupport.ReplaceChunk( start, end, 'string', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a string' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Add_3( self ): # Insert in the middle result_buffer = VimBuffer( 'buffer', contents = [ 'This is a string' ] ) start, end = _BuildLocations( 1, 8, 1, 8 ) vimsupport.ReplaceChunk( start, end, ' not', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is not a string' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Del_1( self ): # Delete from start result_buffer = VimBuffer( 'buffer', contents = [ 'This is a string' ] ) start, end = _BuildLocations( 1, 1, 1, 6 ) vimsupport.ReplaceChunk( start, end, '', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'is a string' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Del_2( self ): # Delete from end result_buffer = VimBuffer( 'buffer', contents = [ 'This is a string' ] ) start, end = _BuildLocations( 1, 10, 1, 18 ) vimsupport.ReplaceChunk( start, end, '', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Del_3( self ): # Delete from middle result_buffer = VimBuffer( 'buffer', contents = [ 'This is not a string' ] ) start, end = _BuildLocations( 1, 9, 1, 13 ) vimsupport.ReplaceChunk( start, end, '', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This is a string' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Unicode_ReplaceUnicodeChars( self ): # Replace Unicode characters. result_buffer = VimBuffer( 'buffer', contents = [ 'This Uniçø∂‰ string is in the middle' ] ) start, end = _BuildLocations( 1, 6, 1, 20 ) vimsupport.ReplaceChunk( start, end, 'Unicode ', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This Unicode string is in the middle' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Unicode_ReplaceAfterUnicode( self ): # Replace ASCII characters after Unicode characters in the line. result_buffer = VimBuffer( 'buffer', contents = [ 'This Uniçø∂‰ string is in the middle' ] ) start, end = _BuildLocations( 1, 30, 1, 43 ) vimsupport.ReplaceChunk( start, end, 'fåke', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'This Uniçø∂‰ string is fåke' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleLine_Unicode_Grown( self ): # Replace ASCII characters after Unicode characters in the line. result_buffer = VimBuffer( 'buffer', contents = [ 'a' ] ) start, end = _BuildLocations( 1, 1, 1, 2 ) vimsupport.ReplaceChunk( start, end, 'å', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'å' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_RemoveSingleLine( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 1, 3, 1 ) vimsupport.ReplaceChunk( start, end, '', result_buffer ) # First line is not affected. AssertBuffersAreEqualAsBytes( [ 'aAa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleToMultipleLines( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 3, 2, 4 ) vimsupport.ReplaceChunk( start, end, 'cccc', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aBcccc', 'aCa' ], result_buffer ) # now make another change to the second line start, end = _BuildLocations( 2, 2, 2, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbF', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aEb', 'bFBcccc', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleToMultipleLines2( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 2, 2, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbFb\nG', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aEb', 'bFb', 'GBa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleToMultipleLines3( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 2, 2, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbFb\nbGb', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aEb', 'bFb', 'bGbBa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleToMultipleLinesReplace( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 1, 2, 1, 4 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbFb\nbGb', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aEb', 'bFb', 'bGb', 'aBa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SingleToMultipleLinesReplace_2( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 1, 4, 1, 4 ) vimsupport.ReplaceChunk( start, end, 'cccc', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAacccc', 'aBa', 'aCa', ], result_buffer ) # now do a subsequent change (insert in the middle of the first line) start, end = _BuildLocations( 1, 2, 1, 4 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbFb\nbGb', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aEb', 'bFb', 'bGbcccc', 'aBa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_MultipleLinesToSingleLine( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCaaaa' ] ) start, end = _BuildLocations( 3, 4, 3, 5 ) vimsupport.ReplaceChunk( start, end, 'dd\ndd', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aBa', 'aCadd', 'ddaa' ], result_buffer ) # make another modification applying offsets start, end = _BuildLocations( 3, 3, 3, 4 ) vimsupport.ReplaceChunk( start, end, 'cccc', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aBa', 'aCccccdd', 'ddaa' ], result_buffer ) # and another, for luck start, end = _BuildLocations( 2, 2, 3, 2 ) vimsupport.ReplaceChunk( start, end, 'E', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aECccccdd', 'ddaa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_MultipleLinesToSameMultipleLines( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa', 'aDe' ] ) start, end = _BuildLocations( 2, 2, 3, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbF', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aEb', 'bFCa', 'aDe' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_MultipleLinesToMoreMultipleLines( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa', 'aDe' ] ) start, end = _BuildLocations( 2, 2, 3, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbFb\nbG', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aEb', 'bFb', 'bGCa', 'aDe' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_MultipleLinesToLessMultipleLines( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa', 'aDe' ] ) start, end = _BuildLocations( 1, 2, 3, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbF', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aEb', 'bFCa', 'aDe' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_MultipleLinesToEvenLessMultipleLines( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa', 'aDe' ] ) start, end = _BuildLocations( 1, 2, 4, 2 ) vimsupport.ReplaceChunk( start, end, 'Eb\nbF', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aEb', 'bFDe' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_SpanBufferEdge( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 1, 1, 1, 3 ) vimsupport.ReplaceChunk( start, end, 'bDb', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'bDba', 'aBa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_DeleteTextInLine( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 2, 2, 3 ) vimsupport.ReplaceChunk( start, end, '', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'aa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_AddTextInLine( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 2, 2, 2 ) vimsupport.ReplaceChunk( start, end, 'bDb', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'abDbBa', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_ReplaceTextInLine( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'aAa', 'aBa', 'aCa' ] ) start, end = _BuildLocations( 2, 2, 2, 3 ) vimsupport.ReplaceChunk( start, end, 'bDb', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'aAa', 'abDba', 'aCa' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_NewlineChunk( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'first line', 'second line' ] ) start, end = _BuildLocations( 1, 11, 2, 1 ) vimsupport.ReplaceChunk( start, end, '\n', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'first line', 'second line' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunk_BeyondEndOfFile( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'first line', 'second line' ] ) start, end = _BuildLocations( 1, 11, 3, 1 ) vimsupport.ReplaceChunk( start, end, '\n', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'first line' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 3 ) ) def test_ReplaceChunk_CursorPosition( self ): result_buffer = VimBuffer( 'buffer', contents = [ 'bar' ] ) start, end = _BuildLocations( 1, 1, 1, 1 ) vimsupport.ReplaceChunk( start, end, 'xyz\nfoo', result_buffer ) AssertBuffersAreEqualAsBytes( [ 'xyz', 'foobar' ], result_buffer ) # Cursor line is 0-based. assert_that( vimsupport.CurrentLineAndColumn(), contains_exactly( 1, 6 ) ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunksInBuffer_SortedChunks( self ): chunks = [ _BuildChunk( 1, 4, 1, 4, '(' ), _BuildChunk( 1, 11, 1, 11, ')' ) ] result_buffer = VimBuffer( 'buffer', contents = [ 'CT<10 >> 2> ct' ] ) vimsupport.ReplaceChunksInBuffer( chunks, result_buffer ) AssertBuffersAreEqualAsBytes( [ 'CT<(10 >> 2)> ct' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunksInBuffer_UnsortedChunks( self ): chunks = [ _BuildChunk( 1, 11, 1, 11, ')' ), _BuildChunk( 1, 4, 1, 4, '(' ) ] result_buffer = VimBuffer( 'buffer', contents = [ 'CT<10 >> 2> ct' ] ) vimsupport.ReplaceChunksInBuffer( chunks, result_buffer ) AssertBuffersAreEqualAsBytes( [ 'CT<(10 >> 2)> ct' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunksInBuffer_LineOverlappingChunks( self ): chunks = [ _BuildChunk( 1, 11, 2, 1, '\n ' ), _BuildChunk( 2, 12, 3, 1, '\n ' ), _BuildChunk( 3, 11, 4, 1, '\n ' ) ] result_buffer = VimBuffer( 'buffer', contents = [ 'first line', 'second line', 'third line', 'fourth line' ] ) vimsupport.ReplaceChunksInBuffer( chunks, result_buffer ) AssertBuffersAreEqualAsBytes( [ 'first line', ' second line', ' third line', ' fourth line' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunksInBuffer_OutdentChunks( self ): chunks = [ _BuildChunk( 1, 1, 1, 5, ' ' ), _BuildChunk( 1, 15, 2, 9, '\n ' ), _BuildChunk( 2, 20, 3, 3, '\n' ) ] result_buffer = VimBuffer( 'buffer', contents = [ ' first line', ' second line', ' third line' ] ) vimsupport.ReplaceChunksInBuffer( chunks, result_buffer ) AssertBuffersAreEqualAsBytes( [ ' first line', ' second line', ' third line' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunksInBuffer_OneLineIndentingChunks( self ): chunks = [ _BuildChunk( 1, 8, 2, 1, '\n ' ), _BuildChunk( 2, 9, 2, 10, '\n ' ), _BuildChunk( 2, 19, 2, 20, '\n ' ) ] result_buffer = VimBuffer( 'buffer', contents = [ 'class {', 'method { statement }', '}' ] ) vimsupport.ReplaceChunksInBuffer( chunks, result_buffer ) AssertBuffersAreEqualAsBytes( [ 'class {', ' method {', ' statement', ' }', '}' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) def test_ReplaceChunksInBuffer_SameLocation( self ): chunks = [ _BuildChunk( 1, 1, 1, 1, 'this ' ), _BuildChunk( 1, 1, 1, 1, 'is ' ), _BuildChunk( 1, 1, 1, 1, 'pure ' ) ] result_buffer = VimBuffer( 'buffer', contents = [ 'folly' ] ) vimsupport.ReplaceChunksInBuffer( chunks, result_buffer ) AssertBuffersAreEqualAsBytes( [ 'this is pure folly' ], result_buffer ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) @patch( 'ycm.vimsupport.VariableExists', return_value = False ) @patch( 'ycm.vimsupport.SetFittingHeightForCurrentWindow' ) @patch( 'vim.command', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.GetBufferNumberForFilename', return_value = 1, new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.BufferIsVisible', return_value = True, new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.OpenFilename' ) @patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock ) @patch( 'vim.eval', new_callable = ExtendedMock ) def test_ReplaceChunks_SingleFile_Open( self, vim_eval, post_vim_message, open_filename, buffer_is_visible, get_buffer_number_for_filename, *args ): single_buffer_name = os.path.realpath( 'single_file' ) chunks = [ _BuildChunk( 1, 1, 2, 1, 'replacement', single_buffer_name ) ] result_buffer = VimBuffer( single_buffer_name, contents = [ 'line1', 'line2', 'line3' ] ) with patch( 'vim.buffers', [ None, result_buffer, None ] ): vimsupport.ReplaceChunks( chunks ) # Ensure that we applied the replacement correctly assert_that( result_buffer.GetLines(), contains_exactly( 'replacementline2', 'line3', ) ) # GetBufferNumberForFilename is called twice: # - once to the check if we would require opening the file (so that we can # raise a warning) # - once whilst applying the changes get_buffer_number_for_filename.assert_has_exact_calls( [ call( single_buffer_name ), call( single_buffer_name ), ] ) # BufferIsVisible is called twice for the same reasons as above buffer_is_visible.assert_has_exact_calls( [ call( 1 ), call( 1 ), ] ) # we don't attempt to open any files open_filename.assert_not_called() qflist = json.dumps( [ { 'bufnr': 1, 'filename': single_buffer_name, 'lnum': 1, 'col': 1, 'text': 'replacement', 'type': 'F' } ] ) # But we do set the quickfix list vim_eval.assert_has_exact_calls( [ call( f'setqflist( { qflist } )' ) ] ) # And it is ReplaceChunks that prints the message showing the number of # changes post_vim_message.assert_has_exact_calls( [ call( 'Applied 1 changes', warning = False ), ] ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) @patch( 'ycm.vimsupport.VariableExists', return_value = False ) @patch( 'ycm.vimsupport.SetFittingHeightForCurrentWindow' ) @patch( 'ycm.vimsupport.GetBufferNumberForFilename', side_effect = [ -1, -1, 1 ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.BufferIsVisible', side_effect = [ False, False, True ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.OpenFilename', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.Confirm', return_value = True, new_callable = ExtendedMock ) @patch( 'vim.eval', return_value = 10, new_callable = ExtendedMock ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_ReplaceChunks_SingleFile_NotOpen( self, vim_command, vim_eval, confirm, post_vim_message, open_filename, buffer_is_visible, get_buffer_number_for_filename, set_fitting_height, variable_exists ): single_buffer_name = os.path.realpath( 'single_file' ) chunks = [ _BuildChunk( 1, 1, 2, 1, 'replacement', single_buffer_name ) ] result_buffer = VimBuffer( single_buffer_name, contents = [ 'line1', 'line2', 'line3' ] ) with patch( 'vim.buffers', [ None, result_buffer, None ] ): vimsupport.ReplaceChunks( chunks ) # We checked if it was OK to open the file confirm.assert_has_exact_calls( [ call( vimsupport.FIXIT_OPENING_BUFFERS_MESSAGE_FORMAT.format( 1 ) ) ] ) # Ensure that we applied the replacement correctly assert_that( result_buffer.GetLines(), contains_exactly( 'replacementline2', 'line3', ) ) # GetBufferNumberForFilename is called 3 times. The return values are set in # the @patch call above: # - once to the check if we would require opening the file (so that we can # raise a warning) (-1 return) # - once whilst applying the changes (-1 return) # - finally after calling OpenFilename (1 return) get_buffer_number_for_filename.assert_has_exact_calls( [ call( single_buffer_name ), call( single_buffer_name ), call( single_buffer_name ), ] ) # BufferIsVisible is called 3 times for the same reasons as above, with the # return of each one buffer_is_visible.assert_has_exact_calls( [ call( -1 ), call( -1 ), call( 1 ), ] ) # We open 'single_file' as expected. open_filename.assert_called_with( single_buffer_name, { 'focus': True, 'fix': True, 'size': 10 } ) # And close it again, then show the quickfix window. vim_command.assert_has_exact_calls( [ call( 'lclose' ), call( 'hide' ), ] ) qflist = json.dumps( [ { 'bufnr': 1, 'filename': single_buffer_name, 'lnum': 1, 'col': 1, 'text': 'replacement', 'type': 'F' } ] ) # And update the quickfix list vim_eval.assert_has_exact_calls( [ call( '&previewheight' ), call( f'setqflist( { qflist } )' ) ] ) # And it is ReplaceChunks that prints the message showing the number of # changes post_vim_message.assert_has_exact_calls( [ call( 'Applied 1 changes', warning = False ), ] ) @patch( 'vim.current.window.cursor', ( 1, 1 ) ) @patch( 'ycm.vimsupport.VariableExists', return_value = False ) @patch( 'ycm.vimsupport.SetFittingHeightForCurrentWindow' ) @patch( 'ycm.vimsupport.GetBufferNumberForFilename', side_effect = [ -1, 1 ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.BufferIsVisible', side_effect = [ False, True ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.OpenFilename', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.Confirm', return_value = True, new_callable = ExtendedMock ) @patch( 'vim.eval', return_value = 10, new_callable = ExtendedMock ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_ReplaceChunks_SingleFile_NotOpen_Silent( self, vim_command, vim_eval, confirm, post_vim_message, open_filename, buffer_is_visible, get_buffer_number_for_filename, set_fitting_height, variable_exists ): # This test is the same as ReplaceChunks_SingleFile_NotOpen_test, but we # pass the silent flag, as used by post-complete actions, and shows the # stuff we _don't_ call in that case. single_buffer_name = os.path.realpath( 'single_file' ) chunks = [ _BuildChunk( 1, 1, 2, 1, 'replacement', single_buffer_name ) ] result_buffer = VimBuffer( single_buffer_name, contents = [ 'line1', 'line2', 'line3' ] ) with patch( 'vim.buffers', [ None, result_buffer, None ] ): vimsupport.ReplaceChunks( chunks, silent=True ) # We didn't check if it was OK to open the file (silent) confirm.assert_not_called() # Ensure that we applied the replacement correctly assert_that( result_buffer.GetLines(), contains_exactly( 'replacementline2', 'line3', ) ) # GetBufferNumberForFilename is called 2 times. The return values are set in # the @patch call above: # - once whilst applying the changes (-1 return) # - finally after calling OpenFilename (1 return) get_buffer_number_for_filename.assert_has_exact_calls( [ call( single_buffer_name ), call( single_buffer_name ), ] ) # BufferIsVisible is called 2 times for the same reasons as above, with the # return of each one buffer_is_visible.assert_has_exact_calls( [ call( -1 ), call( 1 ), ] ) # We open 'single_file' as expected. open_filename.assert_called_with( single_buffer_name, { 'focus': True, 'fix': True, 'size': 10 } ) # And close it again, but don't show the quickfix window vim_command.assert_has_exact_calls( [ call( 'lclose' ), call( 'hide' ), ] ) set_fitting_height.assert_not_called() # But we _don't_ update the QuickFix list vim_eval.assert_has_exact_calls( [ call( '&previewheight' ), ] ) # And we don't print a message either post_vim_message.assert_not_called() @patch( 'vim.current.window.cursor', ( 1, 1 ) ) @patch( 'ycm.vimsupport.GetBufferNumberForFilename', side_effect = [ -1, -1, 1 ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.BufferIsVisible', side_effect = [ False, False, True ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.OpenFilename', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.Confirm', return_value = False, new_callable = ExtendedMock ) @patch( 'vim.eval', return_value = 10, new_callable = ExtendedMock ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_ReplaceChunks_User_Declines_To_Open_File( self, vim_command, vim_eval, confirm, post_vim_message, open_filename, buffer_is_visible, get_buffer_number_for_filename ): # Same as above, except the user selects Cancel when asked if they should # allow us to open lots of (ahem, 1) file. single_buffer_name = os.path.realpath( 'single_file' ) chunks = [ _BuildChunk( 1, 1, 2, 1, 'replacement', single_buffer_name ) ] result_buffer = VimBuffer( single_buffer_name, contents = [ 'line1', 'line2', 'line3' ] ) with patch( 'vim.buffers', [ None, result_buffer, None ] ): vimsupport.ReplaceChunks( chunks ) # We checked if it was OK to open the file confirm.assert_has_exact_calls( [ call( vimsupport.FIXIT_OPENING_BUFFERS_MESSAGE_FORMAT.format( 1 ) ) ] ) # Ensure that buffer is not changed assert_that( result_buffer.GetLines(), contains_exactly( 'line1', 'line2', 'line3', ) ) # GetBufferNumberForFilename is called once. The return values are set in # the @patch call above: # - once to the check if we would require opening the file (so that we can # raise a warning) (-1 return) get_buffer_number_for_filename.assert_has_exact_calls( [ call( single_buffer_name ), ] ) # BufferIsVisible is called once for the above file, which wasn't visible. buffer_is_visible.assert_has_exact_calls( [ call( -1 ), ] ) # We don't attempt to open any files or update any quickfix list or anything # like that open_filename.assert_not_called() vim_eval.assert_not_called() vim_command.assert_not_called() post_vim_message.assert_not_called() @patch( 'vim.current.window.cursor', ( 1, 1 ) ) @patch( 'ycm.vimsupport.GetBufferNumberForFilename', side_effect = [ -1, -1, 1 ], new_callable = ExtendedMock ) # Key difference is here: In the final check, BufferIsVisible returns False @patch( 'ycm.vimsupport.BufferIsVisible', side_effect = [ False, False, False ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.OpenFilename', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.Confirm', return_value = True, new_callable = ExtendedMock ) @patch( 'vim.eval', return_value = 10, new_callable = ExtendedMock ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_ReplaceChunks_User_Aborts_Opening_File( self, vim_command, vim_eval, confirm, post_vim_message, open_filename, buffer_is_visible, get_buffer_number_for_filename ): # Same as above, except the user selects Abort or Quick during the # "swap-file-found" dialog single_buffer_name = os.path.realpath( 'single_file' ) chunks = [ _BuildChunk( 1, 1, 2, 1, 'replacement', single_buffer_name ) ] result_buffer = VimBuffer( single_buffer_name, contents = [ 'line1', 'line2', 'line3' ] ) with patch( 'vim.buffers', [ None, result_buffer, None ] ): assert_that( calling( vimsupport.ReplaceChunks ).with_args( chunks ), raises( RuntimeError, 'Unable to open file: .+single_file\n' 'FixIt/Refactor operation aborted prior to completion. ' 'Your files have not been fully updated. ' 'Please use undo commands to revert the applied changes.' ) ) # We checked if it was OK to open the file confirm.assert_has_exact_calls( [ call( vimsupport.FIXIT_OPENING_BUFFERS_MESSAGE_FORMAT.format( 1 ) ) ] ) # Ensure that buffer is not changed assert_that( result_buffer.GetLines(), contains_exactly( 'line1', 'line2', 'line3', ) ) # We tried to open this file open_filename.assert_called_with( single_buffer_name, { 'focus': True, 'fix': True, 'size': 10 } ) vim_eval.assert_called_with( "&previewheight" ) # But raised an exception before issuing the message at the end post_vim_message.assert_not_called() @patch( 'vim.current.window.cursor', ( 1, 1 ) ) @patch( 'ycm.vimsupport.VariableExists', return_value = False ) @patch( 'ycm.vimsupport.SetFittingHeightForCurrentWindow' ) @patch( 'ycm.vimsupport.GetBufferNumberForFilename', side_effect = [ 22, # first_file (check) -1, # second_file (check) 22, # first_file (apply) -1, # second_file (apply) 19, # second_file (check after open) ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.BufferIsVisible', side_effect = [ True, # first_file (check) False, # second_file (check) True, # first_file (apply) False, # second_file (apply) True, # side_effect (check after open) ], new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.OpenFilename', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock ) @patch( 'ycm.vimsupport.Confirm', return_value = True, new_callable = ExtendedMock ) @patch( 'vim.eval', return_value = 10, new_callable = ExtendedMock ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_ReplaceChunks_MultiFile_Open( self, vim_command, vim_eval, confirm, post_vim_message, open_filename, buffer_is_visible, get_buffer_number_for_filename, set_fitting_height, variable_exists ): # Chunks are split across 2 files, one is already open, one isn't first_buffer_name = os.path.realpath( '1_first_file' ) second_buffer_name = os.path.realpath( '2_second_file' ) chunks = [ _BuildChunk( 1, 1, 2, 1, 'first_file_replacement ', first_buffer_name ), _BuildChunk( 2, 1, 2, 1, 'second_file_replacement ', second_buffer_name ), ] first_file = VimBuffer( first_buffer_name, number = 22, contents = [ 'line1', 'line2', 'line3', ] ) second_file = VimBuffer( second_buffer_name, number = 19, contents = [ 'another line1', 'ACME line2', ] ) vim_buffers = [ None ] * 23 vim_buffers[ 22 ] = first_file vim_buffers[ 19 ] = second_file with patch( 'vim.buffers', vim_buffers ): vimsupport.ReplaceChunks( chunks ) # We checked for the right file names get_buffer_number_for_filename.assert_has_exact_calls( [ call( first_buffer_name ), call( second_buffer_name ), call( first_buffer_name ), call( second_buffer_name ), call( second_buffer_name ), ] ) # We checked if it was OK to open the file confirm.assert_has_exact_calls( [ call( vimsupport.FIXIT_OPENING_BUFFERS_MESSAGE_FORMAT.format( 1 ) ) ] ) # Ensure that buffers are updated assert_that( second_file.GetLines(), contains_exactly( 'another line1', 'second_file_replacement ACME line2', ) ) assert_that( first_file.GetLines(), contains_exactly( 'first_file_replacement line2', 'line3', ) ) # We open '2_second_file' as expected. open_filename.assert_called_with( second_buffer_name, { 'focus': True, 'fix': True, 'size': 10 } ) # And close it again, then show the quickfix window. vim_command.assert_has_exact_calls( [ call( 'lclose' ), call( 'hide' ), ] ) qflist = json.dumps( [ { 'bufnr': 22, 'filename': first_buffer_name, 'lnum': 1, 'col': 1, 'text': 'first_file_replacement ', 'type': 'F' }, { 'bufnr': 19, 'filename': second_buffer_name, 'lnum': 2, 'col': 1, 'text': 'second_file_replacement ', 'type': 'F' } ] ) # And update the quickfix list with each entry vim_eval.assert_has_exact_calls( [ call( '&previewheight' ), call( f'setqflist( { qflist } )' ) ] ) # And it is ReplaceChunks that prints the message showing the number of # changes post_vim_message.assert_has_exact_calls( [ call( 'Applied 2 changes', warning = False ), ] ) @patch( 'vim.command', new_callable=ExtendedMock ) @patch( 'vim.current', new_callable=ExtendedMock ) def test_WriteToPreviewWindow( self, vim_current, vim_command ): vim_current.window.options.__getitem__ = MagicMock( return_value = True ) vimsupport.WriteToPreviewWindow( "test", '' ) vim_command.assert_has_exact_calls( [ call( 'silent! pclose!' ), call( 'silent! pedit! _TEMP_FILE_' ), call( 'silent! wincmd P' ), call( 'silent! wincmd p' ) ] ) vim_current.buffer.__setitem__.assert_called_with( slice( None, None, None ), [ 'test' ] ) vim_current.buffer.options.__setitem__.assert_has_exact_calls( [ call( 'modifiable', True ), call( 'readonly', False ), call( 'buftype', 'nofile' ), call( 'bufhidden', 'wipe' ), call( 'buflisted', False ), call( 'swapfile', False ), call( 'modifiable', False ), call( 'modified', False ), call( 'readonly', True ), ], any_order = True ) @patch( 'vim.command', new_callable=ExtendedMock ) @patch( 'vim.current', new_callable=ExtendedMock ) def test_WriteToPreviewWindow_Mods( self, vim_current, vim_command ): vim_current.window.options.__getitem__ = MagicMock( return_value = True ) vimsupport.WriteToPreviewWindow( "test", 'tab leftabove' ) vim_command.assert_has_exact_calls( [ call( 'silent! pclose!' ), call( 'silent! tab leftabove pedit! _TEMP_FILE_' ), call( 'silent! wincmd P' ), call( 'silent! wincmd p' ) ] ) vim_current.buffer.__setitem__.assert_called_with( slice( None, None, None ), [ 'test' ] ) vim_current.buffer.options.__setitem__.assert_has_exact_calls( [ call( 'modifiable', True ), call( 'readonly', False ), call( 'buftype', 'nofile' ), call( 'bufhidden', 'wipe' ), call( 'buflisted', False ), call( 'swapfile', False ), call( 'modifiable', False ), call( 'modified', False ), call( 'readonly', True ), ], any_order = True ) @patch( 'vim.current' ) def test_WriteToPreviewWindow_MultiLine( self, vim_current ): vim_current.window.options.__getitem__ = MagicMock( return_value = True ) vimsupport.WriteToPreviewWindow( "test\ntest2", '' ) vim_current.buffer.__setitem__.assert_called_with( slice( None, None, None ), [ 'test', 'test2' ] ) @patch( 'vim.command', new_callable=ExtendedMock ) @patch( 'vim.current', new_callable=ExtendedMock ) def test_WriteToPreviewWindow_JumpFail( self, vim_current, vim_command ): vim_current.window.options.__getitem__ = MagicMock( return_value = False ) vimsupport.WriteToPreviewWindow( "test", '' ) vim_command.assert_has_exact_calls( [ call( 'silent! pclose!' ), call( 'silent! pedit! _TEMP_FILE_' ), call( 'silent! wincmd P' ), call( 'redraw' ), call( "echo 'test'" ), ] ) vim_current.buffer.__setitem__.assert_not_called() vim_current.buffer.options.__setitem__.assert_not_called() @patch( 'vim.command', new_callable=ExtendedMock ) @patch( 'vim.current', new_callable=ExtendedMock ) def test_WriteToPreviewWindow_JumpFail_MultiLine( self, vim_current, vim_command ): vim_current.window.options.__getitem__ = MagicMock( return_value = False ) vimsupport.WriteToPreviewWindow( "test\ntest2", '' ) vim_command.assert_has_exact_calls( [ call( 'silent! pclose!' ), call( 'silent! pedit! _TEMP_FILE_' ), call( 'silent! wincmd P' ), call( 'redraw' ), call( "echo 'test'" ), call( "echo 'test2'" ), ] ) vim_current.buffer.__setitem__.assert_not_called() vim_current.buffer.options.__setitem__.assert_not_called() def test_BufferIsVisibleForFilename( self ): visible_buffer = VimBuffer( 'visible_filename', number = 1 ) hidden_buffer = VimBuffer( 'hidden_filename', number = 2 ) with MockVimBuffers( [ visible_buffer, hidden_buffer ], [ visible_buffer ] ): assert_that( vimsupport.BufferIsVisibleForFilename( 'visible_filename' ) ) assert_that( not vimsupport.BufferIsVisibleForFilename( 'hidden_filename' ) ) assert_that( not vimsupport.BufferIsVisibleForFilename( 'another_filename' ) ) def test_CloseBuffersForFilename( self ): current_buffer = VimBuffer( 'some_filename', number = 2 ) other_buffer = VimBuffer( 'some_filename', number = 5 ) with MockVimBuffers( [ current_buffer, other_buffer ], [ current_buffer ] ) as vim: vimsupport.CloseBuffersForFilename( 'some_filename' ) assert_that( vim.buffers, empty() ) @patch( 'vim.command', new_callable = ExtendedMock ) @patch( 'vim.current', new_callable = ExtendedMock ) def test_OpenFilename( self, vim_current, vim_command ): # Options used to open a logfile. options = { 'size': vimsupport.GetIntValue( '&previewheight' ), 'fix': True, 'focus': False, 'watch': True, 'position': 'end' } vimsupport.OpenFilename( __file__, options ) vim_command.assert_has_exact_calls( [ call( f'12split { __file__ }' ), call( f"exec 'au BufEnter :silent! checktime { __file__ }'" ), call( 'silent! normal! Gzz' ), call( 'silent! wincmd p' ) ] ) vim_current.buffer.options.__setitem__.assert_has_exact_calls( [ call( 'autoread', True ), ] ) vim_current.window.options.__setitem__.assert_has_exact_calls( [ call( 'winfixheight', True ) ] ) def test_GetUnsavedAndSpecifiedBufferData_EncodedUnicodeCharsInBuffers( self ): filepath = os.path.realpath( 'filename' ) contents = [ ToBytes( 'abc' ), ToBytes( 'fДa' ) ] vim_buffer = VimBuffer( filepath, contents = contents ) with patch( 'vim.buffers', [ vim_buffer ] ): assert_that( vimsupport.GetUnsavedAndSpecifiedBufferData( vim_buffer, filepath ), has_entry( filepath, has_entry( 'contents', 'abc\nfДa\n' ) ) ) def test_GetBufferFilepath_NoBufferName_UnicodeWorkingDirectory( self ): vim_buffer = VimBuffer( '', number = 42 ) unicode_dir = PathToTestFile( 'uni¢od€' ) with CurrentWorkingDirectory( unicode_dir ): assert_that( vimsupport.GetBufferFilepath( vim_buffer ), equal_to( os.path.join( unicode_dir, '42' ) ) ) # NOTE: Vim returns byte offsets for columns, not actual character columns. # This makes 'ДД' have 4 columns: column 0, column 2 and column 4. @patch( 'vim.current.line', ToBytes( 'ДДaa' ) ) @patch( 'ycm.vimsupport.CurrentColumn', side_effect = [ 4 ] ) def test_TextBeforeCursor_EncodedUnicode( *args ): assert_that( vimsupport.TextBeforeCursor(), equal_to( 'ДД' ) ) # NOTE: Vim returns byte offsets for columns, not actual character columns. # This makes 'ДД' have 4 columns: column 0, column 2 and column 4. @patch( 'vim.current.line', ToBytes( 'aaДД' ) ) @patch( 'ycm.vimsupport.CurrentColumn', side_effect = [ 2 ] ) def test_TextAfterCursor_EncodedUnicode( *args ): assert_that( vimsupport.TextAfterCursor(), equal_to( 'ДД' ) ) @patch( 'vim.current.line', ToBytes( 'fДa' ) ) def test_CurrentLineContents_EncodedUnicode( *args ): assert_that( vimsupport.CurrentLineContents(), equal_to( 'fДa' ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_IntAsUnicode( *args ): assert_that( vimsupport.VimExpressionToPythonType( '123' ), equal_to( 123 ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_IntAsBytes( *args ): assert_that( vimsupport.VimExpressionToPythonType( ToBytes( '123' ) ), equal_to( 123 ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_StringAsUnicode( *args ): assert_that( vimsupport.VimExpressionToPythonType( 'foo' ), equal_to( 'foo' ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_StringAsBytes( *args ): assert_that( vimsupport.VimExpressionToPythonType( ToBytes( 'foo' ) ), equal_to( 'foo' ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_ListPassthrough( *args ): assert_that( vimsupport.VimExpressionToPythonType( [ 1, 2 ] ), equal_to( [ 1, 2 ] ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_ObjectPassthrough( *args ): assert_that( vimsupport.VimExpressionToPythonType( { 1: 2 } ), equal_to( { 1: 2 } ) ) @patch( 'vim.eval', side_effect = lambda x: x ) def test_VimExpressionToPythonType_GeneratorPassthrough( *args ): gen = ( x**2 for x in [ 1, 2, 3 ] ) assert_that( vimsupport.VimExpressionToPythonType( gen ), equal_to( gen ) ) @patch( 'vim.eval', new_callable = ExtendedMock, side_effect = [ None, 2, None ] ) def test_SelectFromList_LastItem( self, vim_eval ): assert_that( vimsupport.SelectFromList( 'test', [ 'a', 'b' ] ), equal_to( 1 ) ) vim_eval.assert_has_exact_calls( [ call( 'inputsave()' ), call( 'inputlist( ["test", "1: a", "2: b"] )' ), call( 'inputrestore()' ) ] ) @patch( 'vim.eval', new_callable = ExtendedMock, side_effect = [ None, 1, None ] ) def test_SelectFromList_FirstItem( self, vim_eval ): assert_that( vimsupport.SelectFromList( 'test', [ 'a', 'b' ] ), equal_to( 0 ) ) vim_eval.assert_has_exact_calls( [ call( 'inputsave()' ), call( 'inputlist( ["test", "1: a", "2: b"] )' ), call( 'inputrestore()' ) ] ) @patch( 'vim.eval', side_effect = [ None, 3, None ] ) def test_SelectFromList_OutOfRange( self, vim_eval ): assert_that( calling( vimsupport.SelectFromList ).with_args( 'test', [ 'a', 'b' ] ), raises( RuntimeError, vimsupport.NO_SELECTION_MADE_MSG ) ) @patch( 'vim.eval', side_effect = [ None, 0, None ] ) def test_SelectFromList_SelectPrompt( self, vim_eval ): assert_that( calling( vimsupport.SelectFromList ).with_args( 'test', [ 'a', 'b' ] ), raises( RuntimeError, vimsupport.NO_SELECTION_MADE_MSG ) ) @patch( 'vim.eval', side_effect = [ None, -199, None ] ) def test_SelectFromList_Negative( self, vim_eval ): assert_that( calling( vimsupport.SelectFromList ).with_args( 'test', [ 'a', 'b' ] ), raises( RuntimeError, vimsupport.NO_SELECTION_MADE_MSG ) ) def test_Filetypes_IntegerFiletype( self ): current_buffer = VimBuffer( 'buffer', number = 1, filetype = '42' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): assert_that( vimsupport.CurrentFiletypes(), contains_exactly( '42' ) ) assert_that( vimsupport.GetBufferFiletypes( 1 ), contains_exactly( '42' ) ) assert_that( vimsupport.FiletypesForBuffer( current_buffer ), contains_exactly( '42' ) ) @patch( 'ycm.vimsupport.VariableExists', return_value = False ) @patch( 'ycm.vimsupport.SearchInCurrentBuffer', return_value = 0 ) @patch( 'vim.current' ) def test_InsertNamespace_insert( self, vim_current, *args ): contents = [ '', 'namespace Taqueria {', '', ' int taco = Math' ] vim_current.buffer = VimBuffer( '', contents = contents ) vim_current.window.cursor = ( 1, 1 ) vimsupport.InsertNamespace( 'System' ) expected_buffer = [ 'using System;', '', 'namespace Taqueria {', '', ' int taco = Math' ] AssertBuffersAreEqualAsBytes( expected_buffer, vim_current.buffer ) @patch( 'ycm.vimsupport.VariableExists', return_value = False ) @patch( 'ycm.vimsupport.SearchInCurrentBuffer', return_value = 2 ) @patch( 'vim.current' ) def test_InsertNamespace_append( self, vim_current, *args ): contents = [ 'namespace Taqueria {', ' using System;', '', ' class Tasty {', ' int taco;', ' List salad = new List' ] vim_current.buffer = VimBuffer( '', contents = contents ) vim_current.window.cursor = ( 1, 1 ) vimsupport.InsertNamespace( 'System.Collections' ) expected_buffer = [ 'namespace Taqueria {', ' using System;', ' using System.Collections;', '', ' class Tasty {', ' int taco;', ' List salad = new List' ] AssertBuffersAreEqualAsBytes( expected_buffer, vim_current.buffer ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_SameFile_SameBuffer_NoSwapFile( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vimsupport.JumpToLocation( os.path.realpath( 'uni¢𐍈d€' ), 2, 5, 'aboveleft', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_SameBuffer_Unmodified( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps belowright edit { target_name }' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_SameFile_NoLineCol( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.cursor = ( 99, 99 ) target_name = os.path.realpath( 'uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, None, None, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 99, 99 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_SameFile_NoLine( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.cursor = ( 99, 99 ) target_name = os.path.realpath( 'uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, None, 1, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 99, 99 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_SameFile_NoCol( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.cursor = ( 99, 99 ) target_name = os.path.realpath( 'uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 1, None, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 99, 99 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_NoLineCol( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.cursor = ( 99, 99 ) target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, None, None, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 99, 99 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps belowright edit { target_name }' ), ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_NoLine( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.cursor = ( 99, 99 ) target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, None, 1, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 99, 99 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps belowright edit { target_name }' ), ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_NoCol( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.window.cursor = ( 99, 99 ) target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 1, None, 'belowright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 99, 99 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps belowright edit { target_name }' ), ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_SameBuffer_Modified_CannotHide( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€', modified = True ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'botright', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps botright split { target_name }' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_SameBuffer_Modified_CanHide( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€', modified = True, bufhidden = "hide" ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'leftabove', 'same-buffer' ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps leftabove edit { target_name }' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', side_effect = [ None, VimError( 'Unknown code' ), None ] ) def test_JumpToLocation_DifferentFile_SameBuffer_SwapFile_Unexpected( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): assert_that( calling( vimsupport.JumpToLocation ).with_args( os.path.realpath( 'different_uni¢𐍈d€' ), 2, 5, 'rightbelow', 'same-buffer' ), raises( VimError, 'Unknown code' ) ) @patch( 'vim.command', new_callable = ExtendedMock, side_effect = [ None, VimError( 'E325' ), None ] ) def test_JumpToLocation_DifferentFile_SameBuffer_SwapFile_Quit( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'topleft', 'same-buffer' ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps topleft edit { target_name }' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock, side_effect = [ None, KeyboardInterrupt, None ] ) def test_JumpToLocation_DifferentFile_SameBuffer_SwapFile_Abort( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'vertical', 'same-buffer' ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps vertical edit { target_name }' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_Split_CurrentTab_NotAlreadyOpened( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) current_window = MagicMock( buffer = current_buffer ) current_tab = MagicMock( windows = [ current_window ] ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ) as vim: vim.current.tabpage = current_tab target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'aboveleft', 'split-or-existing-window' ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps aboveleft split { target_name }' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_Split_CurrentTab_AlreadyOpened( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) different_buffer = VimBuffer( 'different_uni¢𐍈d€' ) current_window = MagicMock( buffer = current_buffer ) different_window = MagicMock( buffer = different_buffer ) current_tab = MagicMock( windows = [ current_window, different_window ] ) with MockVimBuffers( [ current_buffer, different_buffer ], [ current_buffer ] ) as vim: vim.current.tabpage = current_tab vimsupport.JumpToLocation( os.path.realpath( 'different_uni¢𐍈d€' ), 2, 5, 'belowright', 'split-or-existing-window' ) assert_that( vim.current.tabpage, equal_to( current_tab ) ) assert_that( vim.current.window, equal_to( different_window ) ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @WindowsAndMacOnly @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_Split_CurrentTab_AlreadyOpened_Case( self, vim_command ): current_buffer = VimBuffer( 'current_buffer' ) different_buffer = VimBuffer( 'AnotHer_buFfeR' ) current_window = MagicMock( buffer = current_buffer ) different_window = MagicMock( buffer = different_buffer ) current_tab = MagicMock( windows = [ current_window, different_window ] ) with MockVimBuffers( [ current_buffer, different_buffer ], [ current_buffer ] ) as vim: vim.current.tabpage = current_tab vimsupport.JumpToLocation( os.path.realpath( 'anOther_BuffEr' ), 4, 1, 'belowright', 'split-or-existing-window' ) assert_that( vim.current.tabpage, equal_to( current_tab ) ) assert_that( vim.current.window, equal_to( different_window ) ) assert_that( vim.current.window.cursor, equal_to( ( 4, 0 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_Split_AllTabs_NotAlreadyOpened( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'tab', 'split-or-existing-window' ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps tab split { target_name }' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_Split_AllTabs_AlreadyOpened( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) different_buffer = VimBuffer( 'different_uni¢𐍈d€' ) current_window = MagicMock( buffer = current_buffer ) different_window = MagicMock( buffer = different_buffer ) current_tab = MagicMock( windows = [ current_window, different_window ] ) with MockVimBuffers( [ current_buffer, different_buffer ], [ current_buffer ] ) as vim: with patch( 'vim.tabpages', [ current_tab ] ): vimsupport.JumpToLocation( os.path.realpath( 'different_uni¢𐍈d€' ), 2, 5, 'tab', 'split-or-existing-window' ) assert_that( vim.current.tabpage, equal_to( current_tab ) ) assert_that( vim.current.window, equal_to( different_window ) ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_NewOrExistingTab_NotAlreadyOpened( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) with MockVimBuffers( [ current_buffer ], [ current_buffer ] ): target_name = os.path.realpath( 'different_uni¢𐍈d€' ) vimsupport.JumpToLocation( target_name, 2, 5, 'aboveleft vertical', 'new-or-existing-tab' ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( f'keepjumps aboveleft vertical tabedit { target_name }' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'vim.command', new_callable = ExtendedMock ) def test_JumpToLocation_DifferentFile_NewOrExistingTab_AlreadyOpened( self, vim_command ): current_buffer = VimBuffer( 'uni¢𐍈d€' ) different_buffer = VimBuffer( 'different_uni¢𐍈d€' ) current_window = MagicMock( buffer = current_buffer ) different_window = MagicMock( buffer = different_buffer ) current_tab = MagicMock( windows = [ current_window, different_window ] ) with MockVimBuffers( [ current_buffer, different_buffer ], [ current_buffer ] ) as vim: with patch( 'vim.tabpages', [ current_tab ] ): vimsupport.JumpToLocation( os.path.realpath( 'different_uni¢𐍈d€' ), 2, 5, 'belowright tab', 'new-or-existing-tab' ) assert_that( vim.current.tabpage, equal_to( current_tab ) ) assert_that( vim.current.window, equal_to( different_window ) ) assert_that( vim.current.window.cursor, equal_to( ( 2, 4 ) ) ) vim_command.assert_has_exact_calls( [ call( 'normal! m\'' ), call( 'normal! zv' ), call( 'normal! zz' ) ] ) @patch( 'ycm.tests.test_utils.VIM_VERSION', Version( 7, 4, 1578 ) ) def test_VimVersionAtLeast( self ): assert_that( vimsupport.VimVersionAtLeast( '7.3.414' ) ) assert_that( vimsupport.VimVersionAtLeast( '7.4.1578' ) ) assert_that( not vimsupport.VimVersionAtLeast( '7.4.1579' ) ) assert_that( not vimsupport.VimVersionAtLeast( '7.4.1898' ) ) assert_that( not vimsupport.VimVersionAtLeast( '8.1.278' ) )