#!/usr/bin/env python # # Copyright (C) 2013 Strahinja Val Markovic # # 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 webtest import TestApp from .. import ycmd from ..responses import BuildCompletionData from nose.tools import ok_, eq_, with_setup from hamcrest import ( assert_that, has_items, has_entry, contains, contains_string, has_entries ) import bottle bottle.debug( True ) # TODO: Split this file into multiple files. # 'contents' should be just one line of text def RequestDataForFileWithContents( filename, contents = None, filetype = None ): real_contents = contents if contents else '' filetype_to_use = filetype or 'foo' return { 'query': '', 'line_num': 0, 'column_num': 0, 'start_column': 0, 'filetypes': [ filetype_to_use ], 'filepath': filename, 'line_value': real_contents, 'file_data': { filename: { 'contents': real_contents, 'filetypes': [ filetype_to_use ] } } } # TODO: Make the other tests use this helper too instead of BuildCompletionData def CompletionEntryMatcher( insertion_text ): return has_entry( 'insertion_text', insertion_text ) def Setup(): ycmd.SetServerStateToDefaults() @with_setup( Setup ) def GetCompletions_IdentifierCompleter_Works_test(): app = TestApp( ycmd.app ) event_data = RequestDataForFileWithContents( '/foo/bar', 'foo foogoo ba' ) event_data.update( { 'event_name': 'FileReadyToParse', } ) app.post_json( '/event_notification', event_data ) completion_data = RequestDataForFileWithContents( '/foo/bar', 'oo foo foogoo ba' ) completion_data.update( { 'query': 'oo', 'line_num': 0, 'column_num': 2, 'start_column': 0, } ) eq_( [ BuildCompletionData( 'foo' ), BuildCompletionData( 'foogoo' ) ], app.post_json( '/completions', completion_data ).json ) @with_setup( Setup ) def GetCompletions_ClangCompleter_Works_test(): app = TestApp( ycmd.app ) contents = """ struct Foo { int x; int y; char c; }; int main() { Foo foo; foo. } """ filename = '/foo.cpp' completion_data = { 'compilation_flags': ['-x', 'c++'], # 0-based line and column! 'query': '', 'line_num': 10, 'column_num': 6, 'start_column': 6, 'line_value': ' foo.', 'filetypes': ['cpp'], 'filepath': filename, 'file_data': { filename: { 'contents': contents, 'filetypes': ['cpp'] } } } results = app.post_json( '/completions', completion_data ).json assert_that( results, has_items( CompletionEntryMatcher( 'c' ), CompletionEntryMatcher( 'x' ), CompletionEntryMatcher( 'y' ) ) ) @with_setup( Setup ) def GetCompletions_ForceSemantic_Works_test(): app = TestApp( ycmd.app ) completion_data = RequestDataForFileWithContents( 'foo.py', filetype = 'python' ) completion_data.update( { 'force_semantic': True, } ) results = app.post_json( '/completions', completion_data ).json assert_that( results, has_items( CompletionEntryMatcher( 'abs' ), CompletionEntryMatcher( 'open' ), CompletionEntryMatcher( 'bool' ) ) ) @with_setup( Setup ) def GetCompletions_IdentifierCompleter_SyntaxKeywordsAdded_test(): app = TestApp( ycmd.app ) event_data = RequestDataForFileWithContents( '/foo/bar' ) event_data.update( { 'event_name': 'FileReadyToParse', 'syntax_keywords': ['foo', 'bar', 'zoo'] } ) app.post_json( '/event_notification', event_data ) completion_data = RequestDataForFileWithContents( '/foo/bar', 'oo ' ) completion_data.update( { 'query': 'oo', 'line_num': 0, 'column_num': 2, 'start_column': 0, } ) eq_( [ BuildCompletionData( 'foo' ), BuildCompletionData( 'zoo' ) ], app.post_json( '/completions', completion_data ).json ) @with_setup( Setup ) def GetCompletions_UltiSnipsCompleter_Works_test(): app = TestApp( ycmd.app ) event_data = RequestDataForFileWithContents( '/foo/bar' ) event_data.update( { 'event_name': 'BufferVisit', 'ultisnips_snippets': [ {'trigger': 'foo', 'description': 'bar'}, {'trigger': 'zoo', 'description': 'goo'}, ] } ) app.post_json( '/event_notification', event_data ) completion_data = RequestDataForFileWithContents( '/foo/bar', 'oo ' ) completion_data.update( { 'query': 'oo', 'line_num': 0, 'column_num': 2, 'start_column': 0, } ) eq_( [ BuildCompletionData( 'foo', ' bar' ), BuildCompletionData( 'zoo', ' goo' ) ], app.post_json( '/completions', completion_data ).json ) @with_setup( Setup ) def RunCompleterCommand_GoTo_Jedi_ZeroBasedLineAndColumn_test(): app = TestApp( ycmd.app ) contents = """ def foo(): pass foo() """ goto_data = { 'completer_target': 'filetype_default', 'command_arguments': ['GoToDefinition'], 'line_num': 4, 'column_num': 0, 'filetypes': ['python'], 'filepath': '/foo.py', 'file_data': { '/foo.py': { 'contents': contents, 'filetypes': ['python'] } } } # 0-based line and column! eq_( { 'filepath': '/foo.py', 'line_num': 1, 'column_num': 4 }, app.post_json( '/run_completer_command', goto_data ).json ) @with_setup( Setup ) def RunCompleterCommand_GoTo_Clang_ZeroBasedLineAndColumn_test(): app = TestApp( ycmd.app ) contents = """ struct Foo { int x; int y; char c; }; int main() { Foo foo; return 0; } """ filename = '/foo.cpp' goto_data = { 'compilation_flags': ['-x', 'c++'], 'completer_target': 'filetype_default', 'command_arguments': ['GoToDefinition'], 'line_num': 9, 'column_num': 2, 'filetypes': ['cpp'], 'filepath': filename, 'file_data': { filename: { 'contents': contents, 'filetypes': ['cpp'] } } } # 0-based line and column! eq_( { 'filepath': '/foo.cpp', 'line_num': 1, 'column_num': 7 }, app.post_json( '/run_completer_command', goto_data ).json ) @with_setup( Setup ) def DefinedSubcommands_Works_test(): app = TestApp( ycmd.app ) subcommands_data = RequestDataForFileWithContents( '/foo/bar' ) subcommands_data.update( { 'completer_target': 'python', } ) eq_( [ 'GoToDefinition', 'GoToDeclaration', 'GoToDefinitionElseDeclaration' ], app.post_json( '/defined_subcommands', subcommands_data ).json ) @with_setup( Setup ) def DefinedSubcommands_WorksWhenNoExplicitCompleterTargetSpecified_test(): app = TestApp( ycmd.app ) filename = 'foo.py' subcommands_data = { 'filetypes': ['python'], 'filepath': filename, 'file_data': { filename: { 'contents': '', 'filetypes': ['python'] } } } eq_( [ 'GoToDefinition', 'GoToDeclaration', 'GoToDefinitionElseDeclaration' ], app.post_json( '/defined_subcommands', subcommands_data ).json ) @with_setup( Setup ) def Diagnostics_ClangCompleter_ZeroBasedLineAndColumn_test(): app = TestApp( ycmd.app ) contents = """ struct Foo { int x // semicolon missing here! int y; int c; int d; }; """ filename = '/foo.cpp' event_data = { 'event_name': 'FileReadyToParse', 'compilation_flags': ['-x', 'c++'], 'line_num': 0, 'column_num': 0, 'filetypes': ['cpp'], 'filepath': filename, 'file_data': { filename: { 'contents': contents, 'filetypes': ['cpp'] } } } results = app.post_json( '/event_notification', event_data ).json assert_that( results, contains( has_entries( { 'text': contains_string( "expected ';'" ), 'line_num': 2, 'column_num': 7 } ) ) ) @with_setup( Setup ) def GetDetailedDiagnostic_ClangCompleter_Works_test(): app = TestApp( ycmd.app ) contents = """ struct Foo { int x // semicolon missing here! int y; int c; int d; }; """ filename = '/foo.cpp' diag_data = { 'compilation_flags': ['-x', 'c++'], 'line_num': 2, 'column_num': 0, 'filetypes': ['cpp'], 'filepath': filename, 'file_data': { filename: { 'contents': contents, 'filetypes': ['cpp'] } } } event_data = diag_data.copy() event_data.update( { 'event_name': 'FileReadyToParse', } ) app.post_json( '/event_notification', event_data ) results = app.post_json( '/detailed_diagnostic', diag_data ).json assert_that( results, has_entry( 'message', contains_string( "expected ';'" ) ) ) @with_setup( Setup ) def FiletypeCompletionAvailable_Works_test(): app = TestApp( ycmd.app ) request_data = { 'filetypes': ['python'] } ok_( app.post_json( '/filetype_completion_available', request_data ).json ) @with_setup( Setup ) def UserOptions_Works_test(): app = TestApp( ycmd.app ) options = app.get( '/user_options' ).json ok_( len( options ) ) options[ 'foobar' ] = 'zoo' app.post_json( '/user_options', options ) eq_( options, app.get( '/user_options' ).json )