123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257 |
- # -*- coding: utf-8 -*-
- #
- # Copyright (C) 2015 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 <http://www.gnu.org/licenses/>.
- from __future__ import unicode_literals
- from __future__ import print_function
- from __future__ import division
- from __future__ import absolute_import
- from future import standard_library
- standard_library.install_aliases()
- from builtins import * # noqa
- from ycm.test_utils import ExtendedMock, MockVimModule, MockVimCommand
- MockVimModule()
- from ycm import vimsupport
- from nose.tools import eq_
- from hamcrest import assert_that, calling, raises, none, has_entry
- from mock import MagicMock, call, patch
- from ycmd.utils import ToBytes
- import os
- import json
- def ReplaceChunk_SingleLine_Repl_1_test():
- # Replace with longer range
- # 12345678901234567
- result_buffer = [ "This is a string" ]
- start, end = _BuildLocations( 1, 1, 1, 5 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'How long',
- 0,
- 0,
- result_buffer )
- eq_( [ "How long is a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 4 )
- # and replace again, using delta
- start, end = _BuildLocations( 1, 10, 1, 11 )
- ( new_line_offset, new_char_offset ) = vimsupport.ReplaceChunk(
- start,
- end,
- ' piece of ',
- line_offset,
- char_offset,
- result_buffer )
- line_offset += new_line_offset
- char_offset += new_char_offset
- eq_( [ 'How long is a piece of string' ], result_buffer )
- eq_( new_line_offset, 0 )
- eq_( new_char_offset, 9 )
- eq_( line_offset, 0 )
- eq_( char_offset, 13 )
- # and once more, for luck
- start, end = _BuildLocations( 1, 11, 1, 17 )
- ( new_line_offset, new_char_offset ) = vimsupport.ReplaceChunk(
- start,
- end,
- 'pie',
- line_offset,
- char_offset,
- result_buffer )
- line_offset += new_line_offset
- char_offset += new_char_offset
- eq_( ['How long is a piece of pie' ], result_buffer )
- eq_( new_line_offset, 0 )
- eq_( new_char_offset, -3 )
- eq_( line_offset, 0 )
- eq_( char_offset, 10 )
- def ReplaceChunk_SingleLine_Repl_2_test():
- # Replace with shorter range
- # 12345678901234567
- result_buffer = [ "This is a string" ]
- start, end = _BuildLocations( 1, 11, 1, 17 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'test',
- 0,
- 0,
- result_buffer )
- eq_( [ "This is a test" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, -2 )
- def ReplaceChunk_SingleLine_Repl_3_test():
- # Replace with equal range
- # 12345678901234567
- result_buffer = [ "This is a string" ]
- start, end = _BuildLocations( 1, 6, 1, 8 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'be',
- 0,
- 0,
- result_buffer )
- eq_( [ "This be a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 0 )
- def ReplaceChunk_SingleLine_Add_1_test():
- # Insert at start
- result_buffer = [ "is a string" ]
- start, end = _BuildLocations( 1, 1, 1, 1 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'This ',
- 0,
- 0,
- result_buffer )
- eq_( [ "This is a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 5 )
- def ReplaceChunk_SingleLine_Add_2_test():
- # Insert at end
- result_buffer = [ "This is a " ]
- start, end = _BuildLocations( 1, 11, 1, 11 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'string',
- 0,
- 0,
- result_buffer )
- eq_( [ "This is a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 6 )
- def ReplaceChunk_SingleLine_Add_3_test():
- # Insert in the middle
- result_buffer = [ "This is a string" ]
- start, end = _BuildLocations( 1, 8, 1, 8 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- ' not',
- 0,
- 0,
- result_buffer )
- eq_( [ "This is not a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 4 )
- def ReplaceChunk_SingleLine_Del_1_test():
- # Delete from start
- result_buffer = [ "This is a string" ]
- start, end = _BuildLocations( 1, 1, 1, 6 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- '',
- 0,
- 0,
- result_buffer )
- eq_( [ "is a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, -5 )
- def ReplaceChunk_SingleLine_Del_2_test():
- # Delete from end
- result_buffer = [ "This is a string" ]
- start, end = _BuildLocations( 1, 10, 1, 18 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- '',
- 0,
- 0,
- result_buffer )
- eq_( [ "This is a" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, -8 )
- def ReplaceChunk_SingleLine_Del_3_test():
- # Delete from middle
- result_buffer = [ "This is not a string" ]
- start, end = _BuildLocations( 1, 9, 1, 13 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- '',
- 0,
- 0,
- result_buffer )
- eq_( [ "This is a string" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, -4 )
- def ReplaceChunk_RemoveSingleLine_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 2, 1, 3, 1 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, '',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, -1 )
- eq_( char_offset, 0 )
- def ReplaceChunk_SingleToMultipleLines_test():
- result_buffer = [ "aAa",
- "aBa",
- "aCa" ]
- start, end = _BuildLocations( 2, 2, 2, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'Eb\nbF',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa",
- "aEb",
- "bFBa",
- "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 1 )
- eq_( char_offset, 1 )
- # now make another change to the "2nd" line
- start, end = _BuildLocations( 2, 3, 2, 4 )
- ( new_line_offset, new_char_offset ) = vimsupport.ReplaceChunk(
- start,
- end,
- 'cccc',
- line_offset,
- char_offset,
- result_buffer )
- line_offset += new_line_offset
- char_offset += new_char_offset
- eq_( [ "aAa", "aEb", "bFBcccc", "aCa" ], result_buffer )
- eq_( line_offset, 1 )
- eq_( char_offset, 4 )
- def ReplaceChunk_SingleToMultipleLines2_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 2, 2, 2, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'Eb\nbFb\nG',
- 0,
- 0,
- result_buffer )
- expected_buffer = [ "aAa", "aEb", "bFb", "GBa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 2 )
- eq_( char_offset, 0 )
- def ReplaceChunk_SingleToMultipleLines3_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 2, 2, 2, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'Eb\nbFb\nbGb',
- 0,
- 0,
- result_buffer )
- expected_buffer = [ "aAa", "aEb", "bFb", "bGbBa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 2 )
- eq_( char_offset, 2 )
- def ReplaceChunk_SingleToMultipleLinesReplace_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 1, 2, 1, 4 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'Eb\nbFb\nbGb',
- 0,
- 0,
- result_buffer )
- expected_buffer = [ "aEb", "bFb", "bGb", "aBa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 2 )
- eq_( char_offset, 0 )
- def ReplaceChunk_SingleToMultipleLinesReplace_2_test():
- result_buffer = [ "aAa",
- "aBa",
- "aCa" ]
- start, end = _BuildLocations( 1, 2, 1, 4 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'Eb\nbFb\nbGb',
- 0,
- 0,
- result_buffer )
- expected_buffer = [ "aEb",
- "bFb",
- "bGb",
- "aBa",
- "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 2 )
- eq_( char_offset, 0 )
- # now do a subsequent change (insert at end of line "1")
- start, end = _BuildLocations( 1, 4, 1, 4 )
- ( new_line_offset, new_char_offset ) = vimsupport.ReplaceChunk(
- start,
- end,
- 'cccc',
- line_offset,
- char_offset,
- result_buffer )
- line_offset += new_line_offset
- char_offset += new_char_offset
- eq_( [ "aEb",
- "bFb",
- "bGbcccc",
- "aBa",
- "aCa" ], result_buffer )
- eq_( line_offset, 2 )
- eq_( char_offset, 4 )
- def ReplaceChunk_MultipleLinesToSingleLine_test():
- result_buffer = [ "aAa", "aBa", "aCaaaa" ]
- start, end = _BuildLocations( 2, 2, 3, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'E',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa", "aECaaaa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, -1 )
- eq_( char_offset, 1 )
- # make another modification applying offsets
- start, end = _BuildLocations( 3, 3, 3, 4 )
- ( new_line_offset, new_char_offset ) = vimsupport.ReplaceChunk(
- start,
- end,
- 'cccc',
- line_offset,
- char_offset,
- result_buffer )
- line_offset += new_line_offset
- char_offset += new_char_offset
- eq_( [ "aAa", "aECccccaaa" ], result_buffer )
- eq_( line_offset, -1 )
- eq_( char_offset, 4 )
- # and another, for luck
- start, end = _BuildLocations( 3, 4, 3, 5 )
- ( new_line_offset, new_char_offset ) = vimsupport.ReplaceChunk(
- start,
- end,
- 'dd\ndd',
- line_offset,
- char_offset,
- result_buffer )
- line_offset += new_line_offset
- char_offset += new_char_offset
- eq_( [ "aAa", "aECccccdd", "ddaa" ], result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, -2 )
- def ReplaceChunk_MultipleLinesToSameMultipleLines_test():
- result_buffer = [ "aAa", "aBa", "aCa", "aDe" ]
- start, end = _BuildLocations( 2, 2, 3, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'Eb\nbF',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa", "aEb", "bFCa", "aDe" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 1 )
- def ReplaceChunk_MultipleLinesToMoreMultipleLines_test():
- result_buffer = [ "aAa", "aBa", "aCa", "aDe" ]
- start, end = _BuildLocations( 2, 2, 3, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'Eb\nbFb\nbG',
- 0,
- 0,
- result_buffer )
- expected_buffer = [ "aAa", "aEb", "bFb", "bGCa", "aDe" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 1 )
- eq_( char_offset, 1 )
- def ReplaceChunk_MultipleLinesToLessMultipleLines_test():
- result_buffer = [ "aAa", "aBa", "aCa", "aDe" ]
- start, end = _BuildLocations( 1, 2, 3, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'Eb\nbF',
- 0, 0, result_buffer )
- expected_buffer = [ "aEb", "bFCa", "aDe" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, -1 )
- eq_( char_offset, 1 )
- def ReplaceChunk_MultipleLinesToEvenLessMultipleLines_test():
- result_buffer = [ "aAa", "aBa", "aCa", "aDe" ]
- start, end = _BuildLocations( 1, 2, 4, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'Eb\nbF',
- 0, 0, result_buffer )
- expected_buffer = [ "aEb", "bFDe" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, -2 )
- eq_( char_offset, 1 )
- def ReplaceChunk_SpanBufferEdge_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 1, 1, 1, 3 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'bDb',
- 0, 0, result_buffer )
- expected_buffer = [ "bDba", "aBa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 1 )
- def ReplaceChunk_DeleteTextInLine_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 2, 2, 2, 3 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, '',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa", "aa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, -1 )
- def ReplaceChunk_AddTextInLine_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 2, 2, 2, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'bDb',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa", "abDbBa", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 3 )
- def ReplaceChunk_ReplaceTextInLine_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 2, 2, 2, 3 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'bDb',
- 0, 0, result_buffer )
- expected_buffer = [ "aAa", "abDba", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 2 )
- def ReplaceChunk_SingleLineOffsetWorks_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 1, 1, 1, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'bDb',
- 1, 1, result_buffer )
- expected_buffer = [ "aAa", "abDba", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 0 )
- eq_( char_offset, 2 )
- def ReplaceChunk_SingleLineToMultipleLinesOffsetWorks_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 1, 1, 1, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'Db\nE',
- 1, 1, result_buffer )
- expected_buffer = [ "aAa", "aDb", "Ea", "aCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 1 )
- eq_( char_offset, -1 )
- def ReplaceChunk_MultipleLinesToSingleLineOffsetWorks_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 1, 1, 2, 2 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start, end, 'bDb',
- 1, 1, result_buffer )
- expected_buffer = [ "aAa", "abDbCa" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, -1 )
- eq_( char_offset, 3 )
- def ReplaceChunk_MultipleLineOffsetWorks_test():
- result_buffer = [ "aAa", "aBa", "aCa" ]
- start, end = _BuildLocations( 3, 1, 4, 3 )
- ( line_offset, char_offset ) = vimsupport.ReplaceChunk( start,
- end,
- 'bDb\nbEb\nbFb',
- -1,
- 1,
- result_buffer )
- expected_buffer = [ "aAa", "abDb", "bEb", "bFba" ]
- eq_( expected_buffer, result_buffer )
- eq_( line_offset, 1 )
- eq_( char_offset, 1 )
- 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,
- }
- def ReplaceChunksInBuffer_SortedChunks_test():
- chunks = [
- _BuildChunk( 1, 4, 1, 4, '('),
- _BuildChunk( 1, 11, 1, 11, ')' )
- ]
- result_buffer = [ "CT<10 >> 2> ct" ]
- vimsupport.ReplaceChunksInBuffer( chunks, result_buffer, None )
- expected_buffer = [ "CT<(10 >> 2)> ct" ]
- eq_( expected_buffer, result_buffer )
- def ReplaceChunksInBuffer_UnsortedChunks_test():
- chunks = [
- _BuildChunk( 1, 11, 1, 11, ')'),
- _BuildChunk( 1, 4, 1, 4, '(' )
- ]
- result_buffer = [ "CT<10 >> 2> ct" ]
- vimsupport.ReplaceChunksInBuffer( chunks, result_buffer, None )
- expected_buffer = [ "CT<(10 >> 2)> ct" ]
- eq_( expected_buffer, result_buffer )
- class MockBuffer( object ):
- """An object that looks like a vim.buffer object, enough for ReplaceChunk to
- generate a location list"""
- def __init__( self, lines, name, number ):
- self.lines = lines
- self.name = name
- self.number = number
- def __getitem__( self, index ):
- return self.lines[ index ]
- def __len__( self ):
- return len( self.lines )
- def __setitem__( self, key, value ):
- return self.lines.__setitem__( key, value )
- @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.EchoTextVimWidth', new_callable=ExtendedMock )
- @patch( 'vim.eval', new_callable=ExtendedMock )
- @patch( 'vim.command', new_callable=ExtendedMock )
- def ReplaceChunks_SingleFile_Open_test( vim_command,
- vim_eval,
- echo_text_vim_width,
- open_filename,
- buffer_is_visible,
- get_buffer_number_for_filename ):
- chunks = [
- _BuildChunk( 1, 1, 2, 1, 'replacement', 'single_file' )
- ]
- result_buffer = MockBuffer( [
- 'line1',
- 'line2',
- 'line3',
- ], 'single_file', 1 )
- with patch( 'vim.buffers', [ None, result_buffer, None ] ):
- vimsupport.ReplaceChunks( chunks )
- # Ensure that we applied the replacement correctly
- eq_( result_buffer.lines, [
- '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_file', False ),
- call( 'single_file', False ),
- ] )
- # 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()
- # But we do set the quickfix list
- vim_eval.assert_has_exact_calls( [
- call( 'setqflist( {0} )'.format( json.dumps( [ {
- 'bufnr': 1,
- 'filename': 'single_file',
- 'lnum': 1,
- 'col': 1,
- 'text': 'replacement',
- 'type': 'F'
- } ] ) ) ),
- ] )
- vim_command.assert_has_calls( [
- call( 'copen 1' )
- ] )
- # And it is ReplaceChunks that prints the message showing the number of
- # changes
- echo_text_vim_width.assert_has_exact_calls( [
- call( 'Applied 1 changes' ),
- ] )
- @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.EchoTextVimWidth', 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 ReplaceChunks_SingleFile_NotOpen_test( vim_command,
- vim_eval,
- confirm,
- echo_text_vim_width,
- open_filename,
- buffer_is_visible,
- get_buffer_number_for_filename ):
- chunks = [
- _BuildChunk( 1, 1, 2, 1, 'replacement', 'single_file' )
- ]
- result_buffer = MockBuffer( [
- 'line1',
- 'line2',
- 'line3',
- ], 'single_file', 1 )
- 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
- eq_( result_buffer.lines, [
- '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_file', False ),
- call( 'single_file', False ),
- call( 'single_file', False ),
- ] )
- # 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_file', {
- 'focus': True,
- 'fix': True,
- 'size': 10
- } )
- # And close it again, then show the preview window (note, we don't check exact
- # calls because there are other calls which are checked elsewhere)
- vim_command.assert_has_calls( [
- call( 'lclose' ),
- call( 'hide' ),
- call( 'copen 1' ),
- ] )
- # And update the quickfix list
- vim_eval.assert_has_exact_calls( [
- call( '&previewheight' ),
- call( 'setqflist( {0} )'.format( json.dumps( [ {
- 'bufnr': 1,
- 'filename': 'single_file',
- 'lnum': 1,
- 'col': 1,
- 'text': 'replacement',
- 'type': 'F'
- } ] ) ) ),
- ] )
- # And it is ReplaceChunks that prints the message showing the number of
- # changes
- echo_text_vim_width.assert_has_exact_calls( [
- call( 'Applied 1 changes' ),
- ] )
- @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.EchoTextVimWidth',
- 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 ReplaceChunks_User_Declines_To_Open_File_test(
- vim_command,
- vim_eval,
- confirm,
- echo_text_vim_width,
- 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.
- chunks = [
- _BuildChunk( 1, 1, 2, 1, 'replacement', 'single_file' )
- ]
- result_buffer = MockBuffer( [
- 'line1',
- 'line2',
- 'line3',
- ], 'single_file', 1 )
- 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
- eq_( result_buffer.lines, [
- '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_file', False ),
- ] )
- # 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()
- echo_text_vim_width.assert_not_called()
- @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.EchoTextVimWidth',
- 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 ReplaceChunks_User_Aborts_Opening_File_test(
- vim_command,
- vim_eval,
- confirm,
- echo_text_vim_width,
- 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
- chunks = [
- _BuildChunk( 1, 1, 2, 1, 'replacement', 'single_file' )
- ]
- result_buffer = MockBuffer( [
- 'line1',
- 'line2',
- 'line3',
- ], 'single_file', 1 )
- with patch( 'vim.buffers', [ None, result_buffer, None ] ):
- assert_that( calling( vimsupport.ReplaceChunks ).with_args( chunks ),
- raises( RuntimeError,
- 'Unable to open file: single_file\nFixIt/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
- eq_( result_buffer.lines, [
- 'line1',
- 'line2',
- 'line3',
- ] )
- # We tried to open this file
- open_filename.assert_called_with( "single_file", {
- 'focus': True,
- 'fix': True,
- 'size': 10
- } )
- vim_eval.assert_called_with( "&previewheight" )
- # But raised an exception before issuing the message at the end
- echo_text_vim_width.assert_not_called()
- @patch( 'ycm.vimsupport.GetBufferNumberForFilename', side_effect=[
- 22, # first_file (check)
- -1, # another_file (check)
- 22, # first_file (apply)
- -1, # another_file (apply)
- 19, # another_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.EchoTextVimWidth',
- 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 ReplaceChunks_MultiFile_Open_test( vim_command,
- vim_eval,
- confirm,
- echo_text_vim_width,
- open_filename,
- buffer_is_visible,
- get_buffer_number_for_filename ):
- # Chunks are split across 2 files, one is already open, one isn't
- chunks = [
- _BuildChunk( 1, 1, 2, 1, 'first_file_replacement ', '1_first_file' ),
- _BuildChunk( 2, 1, 2, 1, 'second_file_replacement ', '2_another_file' ),
- ]
- first_file = MockBuffer( [
- 'line1',
- 'line2',
- 'line3',
- ], '1_first_file', 22 )
- another_file = MockBuffer( [
- 'another line1',
- 'ACME line2',
- ], '2_another_file', 19 )
- vim_buffers = [ None ] * 23
- vim_buffers[ 22 ] = first_file
- vim_buffers[ 19 ] = another_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( '1_first_file', False ),
- call( '2_another_file', False ),
- call( '1_first_file', False ),
- call( '2_another_file', False ),
- call( '2_another_file', False ),
- ] )
- # 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
- eq_( another_file.lines, [
- 'another line1',
- 'second_file_replacement ACME line2',
- ] )
- eq_( first_file.lines, [
- 'first_file_replacement line2',
- 'line3',
- ] )
- # We open '2_another_file' as expected.
- open_filename.assert_called_with( '2_another_file', {
- 'focus': True,
- 'fix': True,
- 'size': 10
- } )
- # And close it again, then show the preview window (note, we don't check exact
- # calls because there are other calls which are checked elsewhere)
- vim_command.assert_has_calls( [
- call( 'lclose' ),
- call( 'hide' ),
- call( 'copen 2' ),
- ] )
- # And update the quickfix list with each entry
- vim_eval.assert_has_exact_calls( [
- call( '&previewheight' ),
- call( 'setqflist( {0} )'.format( json.dumps( [ {
- 'bufnr': 22,
- 'filename': '1_first_file',
- 'lnum': 1,
- 'col': 1,
- 'text': 'first_file_replacement ',
- 'type': 'F'
- }, {
- 'bufnr': 19,
- 'filename': '2_another_file',
- 'lnum': 2,
- 'col': 1,
- 'text': 'second_file_replacement ',
- 'type': 'F'
- } ] ) ) ),
- ] )
- # And it is ReplaceChunks that prints the message showing the number of
- # changes
- echo_text_vim_width.assert_has_exact_calls( [
- call( 'Applied 2 changes' ),
- ] )
- 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
- }
- @patch( 'vim.command', new_callable=ExtendedMock )
- @patch( 'vim.current', new_callable=ExtendedMock)
- def WriteToPreviewWindow_test( 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( 'swapfile', False ),
- call( 'modifiable', False ),
- call( 'modified', False ),
- call( 'readonly', True ),
- ], any_order = True )
- @patch( 'vim.current' )
- def WriteToPreviewWindow_MultiLine_test( 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 WriteToPreviewWindow_JumpFail_test( 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( "echom '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 WriteToPreviewWindow_JumpFail_MultiLine_test( 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( "echom 'test'" ),
- call( "echom 'test2'" ),
- ] )
- vim_current.buffer.__setitem__.assert_not_called()
- vim_current.buffer.options.__setitem__.assert_not_called()
- def CheckFilename_test():
- assert_that(
- calling( vimsupport.CheckFilename ).with_args( None ),
- raises( RuntimeError, "'None' is not a valid filename" )
- )
- assert_that(
- calling( vimsupport.CheckFilename ).with_args( 'nonexistent_file' ),
- raises( RuntimeError,
- "filename 'nonexistent_file' cannot be opened. "
- "\[Errno 2\] No such file or directory: 'nonexistent_file'" )
- )
- assert_that( vimsupport.CheckFilename( __file__ ), none() )
- def BufferIsVisibleForFilename_test():
- buffers = [
- {
- 'number': 1,
- 'filename': os.path.realpath( 'visible_filename' ),
- 'window': 1
- },
- {
- 'number': 2,
- 'filename': os.path.realpath( 'hidden_filename' ),
- }
- ]
- with patch( 'vim.buffers', buffers ):
- eq_( vimsupport.BufferIsVisibleForFilename( 'visible_filename' ), True )
- eq_( vimsupport.BufferIsVisibleForFilename( 'hidden_filename' ), False )
- eq_( vimsupport.BufferIsVisibleForFilename( 'another_filename' ), False )
- @patch( 'ycm.vimsupport.GetBufferNumberForFilename',
- side_effect = [ 2, 5, -1 ] )
- @patch( 'vim.command',
- side_effect = MockVimCommand,
- new_callable = ExtendedMock )
- def CloseBuffersForFilename_test( vim_command, *args ):
- vimsupport.CloseBuffersForFilename( 'some_filename' )
- vim_command.assert_has_exact_calls( [
- call( 'silent! bwipeout! 2' ),
- call( 'silent! bwipeout! 5' )
- ], any_order = True )
- @patch( 'vim.command', new_callable = ExtendedMock )
- @patch( 'vim.current', new_callable = ExtendedMock )
- def OpenFilename_test( vim_current, vim_command ):
- # Options used to open a logfile
- options = {
- 'size': vimsupport.GetIntValue( '&previewheight' ),
- 'fix': True,
- 'watch': True,
- 'position': 'end'
- }
- vimsupport.OpenFilename( __file__, options )
- vim_command.assert_has_exact_calls( [
- call( '12split {0}'.format( __file__ ) ),
- call( "exec "
- "'au BufEnter <buffer> :silent! checktime {0}'".format( __file__ ) ),
- call( 'silent! normal G zz' ),
- 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 )
- ] )
- @patch( 'ycm.vimsupport.BufferModified', side_effect = [ True ] )
- @patch( 'ycm.vimsupport.FiletypesForBuffer', side_effect = [ [ 'cpp' ] ] )
- def GetUnsavedAndCurrentBufferData_EncodedUnicodeCharsInBuffers_test( *args ):
- mock_buffer = MagicMock()
- mock_buffer.name = os.path.realpath( 'filename' )
- mock_buffer.number = 1
- mock_buffer.__iter__.return_value = [ u'abc', ToBytes( u'fДa' ) ]
- with patch( 'vim.buffers', [ mock_buffer ] ):
- assert_that( vimsupport.GetUnsavedAndCurrentBufferData(),
- has_entry( mock_buffer.name,
- has_entry( u'contents', u'abc\nfДa\n' ) ) )
- # 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 TextBeforeCursor_EncodedUnicode_test( *args ):
- eq_( vimsupport.TextBeforeCursor(), u'ДД' )
- # 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 TextAfterCursor_EncodedUnicode_test( *args ):
- eq_( vimsupport.TextAfterCursor(), u'ДД' )
- @patch( 'vim.current.line', ToBytes( 'fДa' ) )
- def CurrentLineContents_EncodedUnicode_test( *args ):
- eq_( vimsupport.CurrentLineContents(), u'fДa' )
|