greybel-languageserver` is a Language Server for GreyScript that offers a variety of features, including:
- Auto-completion
- Hover tooltips
- Syntax highlighting and more
This language server is compatible with any client that follows the LSP standards.
For an example of how it integrates with a popular editor, take a look at the examples.
greybel-languageserver
supports the following language server protocol (LSP) features:
- Completion: Auto-completion suggestions for code.
- Hover: Displays information about a symbol when you hover over it.
- Color: Color information for syntax highlighting and theming.
- Definition: Navigate to the definition of a symbol.
- Formatter: Automatically format the code according to set rules.
- Signature Help: Shows function or method signatures while typing.
- Document Symbol: Lists all symbols in a document (e.g., functions, classes).
- Workspace Symbol: Search for symbols across the workspace.
- Diagnostic: Provides error, warning, and information diagnostics.
- Semantic Tokens: Enhanced token classification for syntax highlighting and analysis.
npm install -g greybel-languageserver
greybel-languageserver
{
fileExtensions: string; // default: "gs,ms,src"
formatter: boolean; // default: true
autocomplete: boolean; // default: true
hoverdocs: boolean; // default: true
diagnostic: boolean; // default: true
transpiler: {
beautify: {
keepParentheses: boolean; // default: true
indentation: "Tab" | "Whitespace"; // default: "Tab"
indentationSpaces: number; // default: 2
};
};
typeAnalyzer: {
strategy: "Dependency" | "Workspace"; // default: "Dependency"
exclude?: string; // default: undefined
};
}
This section provides a collection of IDEs that implement the greybel-languageserver
.
- VSCode: Visual Studio Code setup for
greybel-languageserver
. - Sublime Text: Instructions for integrating with Sublime Text.
- IntelliJ: Guide for using
greybel-languageserver
with IntelliJ. - Neovim (nvim): Configuration for Neovim users.
- Visual Studio: Learn how to set up a Visual Studio extension using LSP to add support for the GreyScript language in Visual Studio.
Any other IDEs that follow the LSP standards should also work with greybel-languageserver
.
- Create language client file.
import * as path from 'path';
import {
LanguageClient,
LanguageClientOptions,
ServerOptions,
TransportKind
} from 'vscode-languageclient/node';
const serverModule = context.asAbsolutePath(
path.join('node_modules', 'greybel-languageserver', 'index.js')
);
const serverOptions: ServerOptions = {
run: { module: serverModule, transport: TransportKind.ipc }
};
const clientOptions: LanguageClientOptions = {
documentSelector: [{ scheme: 'file', language: 'greyscript' }],
synchronize: {
fileEvents: workspace.createFileSystemWatcher('**/*')
},
diagnosticCollectionName: 'greyscript'
};
const client = new LanguageClient(
'languageServerExample',
'Language Server Example',
serverOptions,
clientOptions
);
client.registerProposedFeatures();
client.start();
- Install the LSP Package from the Sublime Text Package Control.
- Create the following LSP client configuration in your Sublime settings:
{
"show_diagnostics_panel_on_save": 0,
"clients": {
"greyscript": {
"enabled": true,
"command": ["greybel-languageserver", "--stdio"],
"selector": "source.greyscript"
}
},
"semantic_highlighting": true
}
- Create a Sublime syntax file for greyscript. The highlighting will be provided via the semantic provider, so there's no need to add additional patterns here. Use the following configuration:
%YAML 1.2
---
name: greyscript
file_extensions:
- src
scope: source.greyscript
contexts:
main:
- match: '.+'
scope: text.greyscript
To set up greybel-languageserver
in IntelliJ, follow these steps:
- Install greybel-languageserver.
- Install the
LSP4IJ
plugin from the JetBrains Plugin Marketplace. - Go to Languages & Frameworks > Language Servers.
- Click the "+" icon to add a new language server configuration.
- In the Name field, enter
greyscript
. - In the Command field, enter
greybel-languageserver --stdio
. - In the Filename Patterns section:
- Set File Name Pattern to
*.src
. - Set Language Id to
greyscript
.
- Set File Name Pattern to
- Restart IntelliJ.
You should now have greybel-languageserver
set up and ready to use with IntelliJ.
- Add the following configuration to your
init.vim
:
" Install vim-plug if it's not already installed
call plug#begin('~/.vim/plugged')
" Install LSP config plugin
Plug 'neovim/nvim-lspconfig'
call plug#end()
" LSP configuration for greybel-languageserver
lua <<EOF
local configs = require'lspconfig.configs'
local lspconfig = require'lspconfig'
-- Enable debug-level logging
vim.lsp.set_log_level("debug")
if not configs.greybel then
configs.greybel = {
default_config = {
cmd = { "greybel-languageserver", "--stdio" },
filetypes = { "src" },
root_dir = lspconfig.util.root_pattern(".git", vim.fn.getcwd()),
settings = {},
on_attach = function(client, bufnr) -- Optional on_attach function
-- Set up hover keybinding here
vim.api.nvim_buf_set_keymap(bufnr, 'n', 'K', '<cmd>lua vim.lsp.buf.hover()<CR>', { noremap = true, silent = true })
end,
},
}
end
-- Register and start the greybel LSP
lspconfig.greybel.setup{}
EOF
autocmd BufRead,BufNewFile *.src set filetype=src
- Don't forget to run :PlugInstall to install the necessary plugins.
This configuration ensures that greybel-languageserver will be properly integrated into Neovim, and that .src files will be recognized with the correct syntax highlighting and LSP features.
- Begin by following the official Visual Studio Extensibility Tutorial to create a new Visual Studio extension. This will set up the basic structure for the extension project.
- In this step, we define a custom content type for the language we are adding (e.g., GreyScript). This will help Visual Studio identify files based on their extension or content type. Create a new class called ContentTypeDefinitions.cs:
using Microsoft.VisualStudio.LanguageServer.Client;
using Microsoft.VisualStudio.Utilities;
using System.ComponentModel.Composition;
namespace GreyScript
{
internal static class GreyScriptContentDefinition
{
[Export]
[Name("greyscript")]
[BaseDefinition(CodeRemoteContentDefinition.CodeRemoteContentTypeName)]
public static ContentTypeDefinition GreyScriptContentTypeDefinition;
[Export]
[FileExtension(".ms")]
[ContentType("greyscript")]
public static FileExtensionToContentTypeDefinition GreyScriptFileExtensionDefinition;
}
}
- Next, you will create the LanguageClient.cs class that connects Visual Studio to the language server. This class implements ILanguageClient, which is essential for interacting with the LSP. Create a new file called LanguageClient.cs:
using Microsoft.VisualStudio.LanguageServer.Client;
using Microsoft.VisualStudio.Threading;
using Microsoft.VisualStudio.Utilities;
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.Diagnostics;
using System.Threading;
using System.Threading.Tasks;
namespace GreyScript
{
[Export(typeof(ILanguageClient))]
[ContentType("greyscript")]
[RunOnContext(RunningContext.RunOnHost)]
public class GreyScriptLanguageClient : ILanguageClient
{
public event AsyncEventHandler<EventArgs> StartAsync;
public event AsyncEventHandler<EventArgs> StopAsync;
public object InitializationOptions => null;
public IEnumerable<string> FilesToWatch => null;
public bool ShowNotificationOnInitializeFailed => true;
public string Name => "GreyScript Language Client";
public IEnumerable<string> ConfigurationSections => new[] { "greyscript" };
public Task<Connection> ActivateAsync(CancellationToken token)
{
var info = new ProcessStartInfo
{
FileName = @"C:\Users\myUser\AppData\Roaming\npm\greybel-languageserver.cmd",
Arguments = "--stdio",
RedirectStandardInput = true,
RedirectStandardOutput = true,
UseShellExecute = false,
CreateNoWindow = true
};
var process = new Process { StartInfo = info };
if (process.Start())
{
Debug.WriteLine("Language server started successfully.");
return Task.FromResult(new Connection(process.StandardOutput.BaseStream, process.StandardInput.BaseStream));
}
Debug.WriteLine("Failed to start language server.");
return Task.FromResult<Connection>(null);
}
public async Task OnLoadedAsync()
{
if (StartAsync != null)
{
await StartAsync.InvokeAsync(this, EventArgs.Empty);
}
}
public async Task StopServerAsync()
{
if (StopAsync != null)
{
await StopAsync.InvokeAsync(this, EventArgs.Empty);
}
}
public Task OnServerInitializedAsync()
{
return Task.CompletedTask;
}
public Task<InitializationFailureContext> OnServerInitializeFailedAsync(ILanguageClientInitializationInfo initializationState)
{
string message = "GreyScript failed to activate, now we can't test LSP! :(";
string exception = initializationState.InitializationException?.ToString() ?? string.Empty;
message = $"{message}\n {exception}";
var failureContext = new InitializationFailureContext()
{
FailureMessage = message,
};
return Task.FromResult(failureContext);
}
}
}
- At this point, you have a basic framework for integrating a custom language server into Visual Studio. You can customize the content type, server activation, or extend the language client.
Tooltips in greybel-languageserver
can help provide additional context, such as method descriptions, to users. You can contribute your own tooltips by following this workflow:
- Fork and create a pull request (PR) with your changes to the greyscript-meta repository, where the meta descriptions are stored.
- Once your changes are merged, create a separate PR in this repository to update the version of
greybel-languageserver
to include the new meta descriptions.
Additionally, you can define method-specific tooltips directly in the code using comments. This allows for quick, tooltips for individual methods.
// @type Bar
// @property {string} virtualMoo
Bar = {}
Bar.moo = ""
// Hello world
// I am **bold**
// @description Alternative description
// @example test("title", 123)
// @param {string} title - The title of the book.
// @param {string|number} author - The author of the book.
// @return {Bar} - Some info about return
Bar.test = function(test, abc)
print "test"
return self
end function
// @type Foo
Foo = new Bar
// @return {Foo}
Foo.New = function(message)
result = new Foo
return result
end function
myVar = Foo.New
myVar.test // shows defined signature of Bar.test on hover
myVar.virtualMoo // shows virtual property of type string on hover