Metadata-Version: 2.1
Name: super-glass-lsp
Version: 0.4.1
Summary: Generic LSP to parse the output of CLI tools, linters, formatters, etc
License: MIT
Author: Thomas Buckley-Houston
Author-email: tom@tombh.co.uk
Requires-Python: >=3.10,<4.0
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Requires-Dist: mergedeep (>=1.3.4,<2.0.0)
Requires-Dist: parse (>=1.19.0,<2.0.0)
Requires-Dist: pygls (>=0.12.2,<0.13.0)
Requires-Dist: pyyaml (>=6.0,<7.0)
Requires-Dist: single-source (>=0.3.0,<0.4.0)
Description-Content-Type: text/markdown

_🚧 WIP: you're very welcome to try this, but I'm breaking a lot at the moment (October 16th)_

# Super Glass
## Generic LSP Server and/or Pygls Starting Template

<img src="logo.png" align="left" />

This project has 2 goals.

  1. A generic LSP server that parses CLI tools, or indeed any program that outputs to STDOUT, such as  linters, formatters, style checkers, etc and converts their output to LSP-compatible behaviour.
  2. An easily-forkable template to start your own custom LSP server using [Pygls](https://github.com/openlawlibrary/pygls).

Because the heavy-lifting of this language server is done by external tooling (think `pylint`, `jq`, `markdownlint`, etc), there is minimal implementation-specific code in this repo. That is to say that the majority of the code here is applicable to any language server built with [Pygls](https://github.com/openlawlibrary/pygls). Or at the very least, it demonstrates a reasonable starting point. Deleting the `super_glass_lsp/lsp/custom` folder should leave the codebase as close as possible to the minimum starting point for your own custom language server. Then you will also want to rename occurrences of `[C|c]ustom` to your own language server's name.

## Installation

`pip install super-glass-lsp`

## Usage

### Quickstart
Once you've installed the language server and [set it up in your editor](https://github.com/tombh/super-glass#editor-setups), it should be as easy as this to add new features (this is YAML, but your editor likely has its own config format):
```yaml
# This is jsut an ID, so can be anything. Internally it's important so that you can
# override existing configs (either the bundled defaults, or configs you have
# created elsewhere): all configs with the same ID are automaticallly merged. 
fuzzy_similar_words_completion:

  # This is the part of the language server to which the `command` will apply.
  # The other currently supported features are: `diagnostic`.
  lsp_feature: completion
  
  # This is the external command which will be triggered and parsed for every
  # invocation of the feature. In the case of completions, editors will generally
  # trigger it for _every_ character change, or even every key press. So be
  # careful not to make this too expensive.
  #
  # Default behaviour is to pipe the entire contents of the file into the command.
  # This can be overriden with `piped: false`. In which case you will likely want
  # to manually do something with the file. You can access its path with the `{file}`
  # token. Eg; `command: "cat {file} | tr ..."`.
  #
  # This particular command first breaks up the file into a list of words, which are
  # then piped into a fuzzy finder, which then queries the list with the particular
  # word currently under your cursor in the editor. Finally the results of the fuzzy
  # search are deduplicated (with `uniq`).
  #
  # The command is run in a shell, so all the tools from your own machine are available.
  command: "tr -cs '[:alnum:]' '\n' | fzf --filter='{word}' | uniq"
```

### Configuration

The server comes with a lot of [defaults](super_glass_lsp/config.default.yaml). To enable a particular tool simple provide the `enabled: true` field for that tool. For example:
```yaml
# This is YAML, but should be whatever format your editor's config is
initialization_options:
  configs:
    jqlint:
      enabled: true
```

TODO:
* [ ] Explain all the fields and tokens for each LSP feature
* [ ] Remember to describe the format array lines priorities

## Editor Setups

Because this is a generic language server, the filetype/language that the server applies to varies depending on the config you've setup. It would be a bad idea for a generic language server to tell an editor that it wants to connect with every possible filetype/language (although this can be enabled on a per tool basis with the `language_ids: ["*"]` setting). Instead, it is better that you manually inform your editor which filetypes/languages this generic server should be enabled for. How that is done is unique to each editor's config, I've tried to include examples for each one.

<details>
<summary>Neovim Lua (vanilla Neovim without `lspconfig`)</summary>

  Since this project is very beta, we're not yet submitting this language server to the LSP Config plugin (the defacto way to add new language servers). Therefore, for now, we have to use Neovim's vanilla LSP setup (which has actually simplified a lot recently).

  ```lua
  vim.api.nvim_create_autocmd({ "BufEnter" }, {
    -- NB: You must remember to manually put the file extension pattern matchers for each LSP filetype
    pattern = { "*" },
    callback = function()
      vim.lsp.start({
        name = "super-glass",
        cmd = { "super-glass-lsp" },
        root_dir = vim.fs.dirname(vim.fs.find({ ".git" }, { upward = true })[1]),
        init_options = {
          configs = {
            fuzzy_buffer_tokens = {
              lsp_feature = "completion",
              command = "tr -cs '[:alnum:]' '\n' | fzf --filter='{word}' | uniq",
            },
          }
        },
      })
    end,
  })
  ```
</details>

<details>
<summary>Vim (`vim-lsp`)</summary>

  ```vim
  augroup LspCLITools
  au!
  autocmd User lsp_setup call lsp#register_server({
      \ 'name': 'super-glass',
      \ 'cmd': {server_info->['super-glass-lsp', '--logfile', 'path/to-logfile']},
      \ 'allowlist': ['vim', 'eruby', 'markdown', 'yaml'],
      \ 'initialization_options': { "configs":
      \   { "fuzzy_buffer_tokens": {
      \       "lsp_feature": "completion",
      \       "command": "tr -cs '[:alnum:]' '\n' | fzf --filter='{word}' | uniq",
      \     }
      \   }
      \ }})
  augroup END
  ```
</details>

<details>
<summary>Neovim (`lspconfig`) TBC</summary>

  Once we're stable, we'll submit ourselves for inclusion.
</details>

<details>
<summary>Emacs (`lsp-mode`)</summary>


  ```
  (make-lsp-client :new-connection
  (lsp-stdio-connection
    `(,(executable-find "super-glass-lsp") "--logfile" "path/to/logs"))
    :activation-fn (lsp-activate-on "json")
    :initialization-options ; TODO: I'm not an Emacs user, how do we provide these options?
    :server-id 'super-glass-lsp')))
  ```
</details>

<details>
<summary>Emacs (`eglot`) TBC</summary>
  
  Once we're stable, we'll submit ourselves for inclusion.
</details>

<details>
<summary>VSCode TBC</summary>
  
  Can we copy EFM's VSCode extension?
  https://github.com/Matts966/efm-langserver-vscode
</details>


## Testing

Uses [@alcarney](https://github.com/alcarney)'s [pytest-lsp module](https://github.com/alcarney/lsp-devtools/tree/develop/lib/pytest-lsp) for end-to-end testing.

`poetry run python -m pytest`

## Acknowledgements

This projects takes a lot of inspiration from [@alcarney](https://github.com/alcarney)'s fantastic Sphinx/RST LSP server [Esbonio](https://github.com/swyddfa/esbonio). 

Logo is from [a sticker I found on Amazon](https://www.amazon.com/-/es/Superman-S-Adhesivo-reflectante-plateado/dp/B00PEZKHV8), obviously want a proper logo before I publish.

## Other generic LSP servers

* https://github.com/iamcco/diagnostic-languageserver
* https://github.com/mattn/efm-langserver
* https://github.com/jose-elias-alvarez/null-ls.nvim (Neovim only)

