# frozen_string_literal: true require "benchmark" module ThemeCheck module LanguageServer class Handler include URIHelper SERVER_INFO = { name: $PROGRAM_NAME, version: ThemeCheck::VERSION, } # https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification/#fileOperationFilter FILE_OPERATION_FILTER = { filters: [{ scheme: 'file', pattern: { glob: '**/*', }, }], } CAPABILITIES = { completionProvider: { triggerCharacters: ['.', '{{ ', '{% '], context: true, }, codeActionProvider: { codeActionKinds: CodeActionProvider.all.map(&:kind), resolveProvider: false, workDoneProgress: false, }, documentLinkProvider: true, executeCommandProvider: { workDoneProgress: false, commands: ExecuteCommandProvider.all.map(&:command), }, textDocumentSync: { openClose: true, change: TextDocumentSyncKind::FULL, willSave: false, save: true, }, workspace: { fileOperations: { didCreate: FILE_OPERATION_FILTER, didDelete: FILE_OPERATION_FILTER, willRename: FILE_OPERATION_FILTER, }, }, } def initialize(bridge) @bridge = bridge end def on_initialize(id, params) @root_path = root_path_from_params(params) # Tell the client we don't support anything if there's no rootPath return @bridge.send_response(id, { capabilities: {} }) if @root_path.nil? @client_capabilities = ClientCapabilities.new(params.dig(:capabilities) || {}) @configuration = Configuration.new(@bridge, @client_capabilities) @bridge.supports_work_done_progress = @client_capabilities.supports_work_done_progress? @storage = in_memory_storage(@root_path) @diagnostics_manager = DiagnosticsManager.new @completion_engine = CompletionEngine.new(@storage, @bridge) @document_link_engine = DocumentLinkEngine.new(@storage) @diagnostics_engine = DiagnosticsEngine.new(@storage, @bridge, @diagnostics_manager) @execute_command_engine = ExecuteCommandEngine.new @execute_command_engine << CorrectionExecuteCommandProvider.new(@storage, @bridge, @diagnostics_manager) @execute_command_engine << RunChecksExecuteCommandProvider.new( @diagnostics_engine, @storage, config_for_path(@root_path), @configuration, ) @code_action_engine = CodeActionEngine.new(@storage, @diagnostics_manager) @bridge.send_response(id, { capabilities: CAPABILITIES, serverInfo: SERVER_INFO, }) end def on_initialized(_id, _params) return unless @configuration @configuration.fetch @configuration.register_did_change_capability ShopifyLiquid::SourceManager.download_or_refresh_files end def on_shutdown(id, _params) @bridge.send_response(id, nil) end def on_exit(_id, _params) close! end def on_text_document_did_open(_id, params) relative_path = relative_path_from_text_document_uri(params) @storage.write(relative_path, text_document_text(params), text_document_version(params)) analyze_and_send_offenses(text_document_uri(params)) if @configuration.check_on_open? end def on_text_document_did_change(_id, params) relative_path = relative_path_from_text_document_uri(params) @storage.write(relative_path, content_changes_text(params), text_document_version(params)) analyze_and_send_offenses(text_document_uri(params), only_single_file: true) if @configuration.check_on_change? end def on_text_document_did_close(_id, params) relative_path = relative_path_from_text_document_uri(params) begin file_system_content = Pathname.new(text_document_uri(params)).read(mode: 'rb', encoding: 'UTF-8') # On close, the file system becomes the source of truth @storage.write(relative_path, file_system_content, nil) # the file no longer exists because either the user deleted it, or the user renamed it. rescue Errno::ENOENT @storage.remove(relative_path) ensure @diagnostics_engine.clear_diagnostics(relative_path) if @configuration.only_single_file? end end def on_text_document_did_save(_id, params) analyze_and_send_offenses(text_document_uri(params)) if @configuration.check_on_save? end def on_text_document_document_link(id, params) relative_path = relative_path_from_text_document_uri(params) @bridge.send_response(id, @document_link_engine.document_links(relative_path)) end def on_text_document_completion(id, params) relative_path = relative_path_from_text_document_uri(params) line = params.dig(:position, :line) col = params.dig(:position, :character) @bridge.send_response(id, @completion_engine.completions(relative_path, line, col)) end def on_text_document_code_action(id, params) absolute_path = text_document_uri(params) start_position = range_element(params, :start) end_position = range_element(params, :end) only_code_action_kinds = params.dig(:context, :only) || [] @bridge.send_response(id, @code_action_engine.code_actions( absolute_path, start_position, end_position, only_code_action_kinds, )) end def on_workspace_did_create_files(_id, params) paths = params[:files] &.map { |file| file[:uri] } &.map { |uri| file_path(uri) } return unless paths paths.each do |path| relative_path = @storage.relative_path(path) file_system_content = Pathname.new(path).read(mode: 'rb', encoding: 'UTF-8') @storage.write(relative_path, file_system_content, nil) end end def on_workspace_did_delete_files(_id, params) absolute_paths = params[:files] &.map { |file| file[:uri] } &.map { |uri| file_path(uri) } return unless absolute_paths absolute_paths.each do |path| relative_path = @storage.relative_path(path) @storage.remove(relative_path) end analyze_and_send_offenses(absolute_paths) end # We're using workspace/willRenameFiles here because we want this to run # before textDocument/didOpen and textDocumetn/didClose of the files # (which might trigger another theme analysis). def on_workspace_will_rename_files(id, params) relative_paths = params[:files] &.map { |file| [file[:oldUri], file[:newUri]] } &.map { |(old_uri, new_uri)| [relative_path_from_uri(old_uri), relative_path_from_uri(new_uri)] } return @bridge.send_response(id, nil) unless relative_paths relative_paths.each do |(old_path, new_path)| @storage.write(new_path, @storage.read(old_path), nil) @storage.remove(old_path) end @bridge.send_response(id, nil) absolute_paths = relative_paths.flatten(2).map { |p| @storage.path(p) } analyze_and_send_offenses(absolute_paths) end def on_workspace_execute_command(id, params) @bridge.send_response(id, @execute_command_engine.execute( params[:command], params[:arguments], )) end def on_workspace_did_change_configuration(_id, _params) @configuration.fetch(force: true) end private def in_memory_storage(root) config = config_for_path(root) # Make a real FS to get the files from the snippets folder fs = ThemeCheck::FileSystemStorage.new( config.root, ignored_patterns: config.ignored_patterns ) # Turn that into a hash of buffers files = fs.files .map { |fn| [fn, fs.read(fn)] } .to_h VersionedInMemoryStorage.new(files, config.root) end def text_document_uri(params) file_path(params.dig(:textDocument, :uri)) end def relative_path_from_uri(uri) @storage.relative_path(file_path(uri)) end def relative_path_from_text_document_uri(params) @storage.relative_path(text_document_uri(params)) end def root_path_from_params(params) root_uri = params[:rootUri] root_path = params[:rootPath] if root_uri file_path(root_uri) elsif root_path root_path end end def text_document_text(params) params.dig(:textDocument, :text) end def text_document_version(params) params.dig(:textDocument, :version) end def content_changes_text(params) params.dig(:contentChanges, 0, :text) end def config_for_path(path_or_paths) path = path_or_paths.is_a?(Array) ? path_or_paths[0] : path_or_paths root = ThemeCheck::Config.find(path) || @root_path ThemeCheck::Config.from_path(root) end def analyze_and_send_offenses(absolute_path_or_paths, only_single_file: nil) @diagnostics_engine.analyze_and_send_offenses( absolute_path_or_paths, config_for_path(absolute_path_or_paths), only_single_file: only_single_file.nil? ? @configuration.only_single_file? : only_single_file ) end def range_element(params, start_or_end) [ params.dig(:range, start_or_end, :line), params.dig(:range, start_or_end, :character), ] end def log(message) @bridge.log(message) end def close! raise DoneStreaming end end end end