psc-ide/PROTOCOL.md
Communication with purs ide server is via a JSON protocol over a TCP connection:
the server listens on a particular (configurable) port, and will accept a single line
of JSON input in the format described below, terminated by a newline, before giving
a JSON response and closing the connection.
The purs ide client command can be used as a wrapper for the TCP connection, but
otherwise behaves the same, accepting a line of JSON on stdin and exiting after
giving a result on stdout.
The result needs to be unwrapped from the "wrapper" which separates success from failure:
{
"resultType": "success|error",
"result": Result|Error
}
The load command "loads" the requested modules into the server for completion
and type info. If the params object is left off, the load command will try
to detect all the compiled modules in your project and load them.
Params:
modules :: (optional) [ModuleName]: A list of modules to load.
psc-ide-server will try to parse all the declarations in these modules{
"command": "load",
"params": (optional) {
"modules": (optional)["Module.Name1", "Module.Name2"]
}
}
Result:
The Load Command returns a string with a summary about the loading process.
The type command looks up the type for a given identifier. It also returns the
definition position, if it can be found in the passed source files.
Params:
search :: String: The identifier to look for. Only matches on equality.filters :: (optional) [Filter]: These filters will be applied before looking for the
identifier. These filters get combined with AND, so a candidate must match ALL
of them to be eligible.currentModule :: (optional) String: see Complete command{
"command": "type",
"params": {
"search": "filterM",
"filters": [{..}],
"currentModule": "Main"
}
}
Result: The possible types are returned in the same format as completions
The complete command looks up possible completions/corrections.
Params:
filters :: [Filter]: The same as for the type command. A candidate must
match all filters.
matcher :: (optional) Matcher: The strategy used for matching candidates
after filtering. Results are scored internally and will be returned in the
descending order where the nth element is better then the n+1-th.
If no matcher is given every candidate, that passes the filters, is returned
in no particular order.
currentModule :: (optional) String: The current modules name. Allows you
to see module-private functions after a successful rebuild. If it matches
with the rebuild cache non-exported modules will also be completed. You can
fill the rebuild cache by using the "Rebuild" command.
options :: (optional) CompletionOptions: The CompletionOptions to apply to
the completion results
{
"command": "complete",
"params": {
"filters": [{..}, {..}],
"matcher": {..},
"currentModule": "Main",
"options": {
"maxResults": 50,
"groupReexports": true
}
}
}
Result:
The following format is returned as the Result:
The definedAt, documentation, as well as the declarationType field might
be null if they couldn't be extracted from a source file. See the
Declaration Type Filter further down for all
possible values of declaration types and how to use this information.
[
{
"module": "Data.Array",
"identifier": "filter",
"type": "forall a. (a -> Boolean) -> Array a -> Array a",
"expandedType": "forall a. (a -> Boolean) -> Array a -> Array a",
"definedAt":
{
"name": "/path/to/file",
"start": [1, 3],
"end": [3, 1]
},
"documentation": "A filtering function",
"exportedFrom": ["Data.Array"],
"declarationType": "value",
}
]
The CaseSplit command takes a line of source code, an area in that line of code
and replaces it with all patterns for a given type. The parameter annotations
is used to turn type annotations on or off for the constructor fields.
{
"command": "caseSplit",
"params": {
"line": "elem a as",
"begin": 8,
"end": 10,
"annotations": true,
"type": "List"
}
}
Result:
The following format is returned as the Result:
[
"elem a Nil",
"elem a (Cons (_ :: a) (_ :: List a))"
]
You should then be able to replace the affected line of code in the editor with the new suggestions.
The AddClause command takes a typedeclaration and generates a function template for the given type.
The annotations option turns type annotations on or off for the function arguments.
{
"command": "addClause",
"params": {
"line": "elem :: forall a. (Eq a) => a -> List a",
"annotations": true
}
}
Result:
The following format is returned as the Result:
[
"elem :: forall a. (Eq a) => a -> List a",
"elem ( _ :: a) = ?elem"
]
You should then be able to replace the affected line of code in the editor with the new suggestions.
The Usages command accepts a triplet of modulename, namespace, and identifier, which uniquely identify a declaration and returns all usages of that identifier in all loaded files. Note that we use the parsed source files, so you need to pass source globs at startup to use this command.
{
"command": "usages",
"params": {
"module": "Data.Array",
"namespace": "value|type|kind",
"identifier": "filter"
}
}
Result:
The following format is returned as the Result:
[ { "name": "/path/to/file"
, "start": [1, 3]
, "end": [3, 1]
}
, { "name": "/path/to/file"
, "start": [5, 6]
, "end": [5, 8]
}
]
For now all of the import related commands work with a file on the filesystem.
You can specify it with the file parameter.
If you supply the optional outfile parameter, the output will be written to
that file, and an info message will be returned from the client.
If you don't supply outfile, the server responds with a list of strings which,
when inserted into a file linewise create the module with the applied changes.
Arguments:
file :: Stringoutfile :: Maybe Stringfilters :: Maybe [Filter]Example:
{
"command": "import",
"params": {
"file": "/home/creek/Documents/chromacannon/src/Main.purs",
"outfile": "/home/creek/Documents/chromacannon/src/Main.purs",
"filters": [{
"filter": "modules",
"params": {
"modules": ["My.Module"]
}
}],
"importCommand": {
"yadda": "yadda"
}
}
}
addImplicitImportThis command just adds an unqualified import for the given modulename.
Arguments:
module :: StringExample:
{
"command": "import",
"params": {
"file": "/home/creek/Documents/chromacannon/src/Main.purs",
"importCommand": {
"importCommand": "addImplicitImport",
"module": "Data.Array.LOL"
}
}
}
addQualifiedImportThis command adds an import for the given modulename and qualifier.
Arguments:
module :: Stringqualifier :: StringExample:
{
"command": "import",
"params": {
"file": "/home/creek/Documents/chromacannon/src/Main.purs",
"importCommand": {
"importCommand": "addQualifiedImport",
"module": "Data.Array",
"qualifier": "Array"
}
}
}
addImportThis command takes an identifier and searches the currently loaded modules for it. If it finds no matches it responds with an Error. If it finds exactly one match it adds the import and returns. If it finds more than one match it responds with a list of the found matches as completions like the complete command.
You can also supply a list of filters like the ones for completion. These are
specified as part of the top level command rather than within the importCommand.
This way you can narrow down the search to a certain module and resolve the case in which
more then one match was found.
Arguments:
identifier :: Stringqualifier :: String (optional)Example:
{
"command": "import",
"params": {
"file": "/home/creek/Documents/chromacannon/src/Demo.purs",
"outfile": "/home/creek/Documents/chromacannon/src/Demo.purs",
"importCommand": {
"importCommand": "addImport",
"identifier": "bind"
}
}
}
Example with qualifier and filter:
{
"command": "import",
"params": {
"file": "/home/creek/Documents/chromacannon/src/Demo.purs",
"outfile": "/home/creek/Documents/chromacannon/src/Demo.purs",
"importCommand": {
"importCommand": "addImport",
"identifier": "length",
"qualifier": "Array"
},
"filters": [{
"filter": "modules",
"params": {
"modules": ["Data.Array"]
}
}]
}
}
The rebuild command provides a fast rebuild for a single module. It doesn't
recompile the entire project though. All the modules dependencies need to be
loaded. A successful rebuild will be stored to allow for completions of private
identifiers.
Arguments:
file :: String the path to the module to rebuild or the complete
source code of the module to be compiled prefixed with data:actualFile :: Maybe String Specifies the path to be used for location
information and parse errors. This is useful in case a temp file is used as
the source for a rebuild.codegen :: Maybe [String] Specified the codegen targets the
rebuild should produce. Uses the same target names as the command
line compiler. Defaults to just JS output{
"command": "rebuild",
"params": {
"file": "/path/to/file.purs",
"actualFile": "/path/to/actualFile.purs",
"codegen": ["js", "corefn"]
}
}
Result
In the Success case you get a list of warnings in the compilers json format.
In the Error case you get the errors in the compilers json format
This command will be removed in the next breaking release after 0.13, use the completion command with a filter for modules instead.
list of type loadedModules lists all loaded modules (This means they can be searched for completions etc)
{
"command": "list",
"params": {
"type": "loadedModules"
}
}
The list loadedModules command returns a list of strings.
list of type availableModules lists all available modules. (This basically
means the contents of the output/ folder.))
{
"command": "list",
"params": {
"type": "availableModules"
}
}
The list availableModules command returns a list of strings.
The list command can also list the imports for a given file.
{
"command": "list",
"params": {
"type": "import",
"file": "/home/kritzcreek/Documents/psc-ide/examples/Main.purs"
}
}
The list import command returns the parse module name as well as a list of imports like so:
{
"moduleName": "MyModule",
"imports": [Import]
}
The different kind of imports are returned like so:
Implicit Import (import Data.Array):
{
"module": "Data.Array",
"importType": "implicit"
}
Implicit qualified Import (import Data.Array as A):
{
"module": "Data.Array",
"importType": "implicit",
"qualifier": "A"
}
Explicit Import (import Data.Array (filter, filterM, join)):
{
"module": "Data.Array",
"importType": "explicit",
"identifiers": ["filter", "filterM", "join"]
}
Explicit qualified Import (import Data.Array (filter, filterM, join) as A):
{
"module": "Data.Array",
"importType": "explicit",
"identifiers": ["filter", "filterM", "join"],
"qualifier": "A"
}
Hiding Import (import Data.Array hiding (filter, filterM, join)):
{
"module": "Data.Array",
"importType": "hiding",
"identifiers": ["filter", "filterM", "join"]
}
Qualified Hiding Import (import Data.Array hiding (filter, filterM, join) as A):
{
"module": "Data.Array",
"importType": "hiding",
"identifiers": ["filter", "filterM", "join"],
"qualifier": "A"
}
cwd returns the working directory of the server(should be your project root).
quit quits the server.
reset resets all loaded modules.
{
"command": "cwd|quit|reset"
}
Result: These commands return strings.
The Exact filter only keeps identifiers that are equal to the search term.
{
"filter": "exact",
"params": {
"search": "filterM"
}
}
The Prefix filter keeps identifiers/modules/data declarations that are prefixed by the search term.
{
"filter": "prefix",
"params": {
"search": "filt"
}
}
The Module filter only keeps identifiers that appear in the listed modules.
{
"filter": "modules",
"params": {
"modules": ["My.Module"]
}
}
The Dependency filter only keeps identifiers that appear in the listed module or are brought into scope by any of its imports.
The module text is provided, though only the portion up until the end of the import section need be provided.
Parameters:
moduleText :: Stringqualifier :: String (optional){
"filter": "dependencies",
"params": {
"moduleText": "module My.Module where\nimport Foo as F\n",
"qualifier": "F"
}
}
The Namespace filter only keeps identifiers that appear in the listed namespaces.
Valid namespaces are value, type and kind.
{
"filter": "namespace",
"params": {
"namespaces": ["value", "type", "kind"]
}
}
A filter which allows to filter type declarations. Valid type declarations are
value, type, synonym, dataconstructor, typeclass, valueoperator,
typeoperator, kind, and module.
{
"filter": "declarations",
"params":
[ "value"
, "type"
, "synonym"
, "dataconstructor"
, "typeclass"
, "valueoperator"
, "typeoperator"
, "kind"
, "module"
]
}
Matches any occurrence of the search string with intersections
The scoring measures how far the matches span the string, where closer is better. The matches then get sorted with highest score first.
Examples:
{
"matcher": "flex",
"params": {
"search": "filt"
}
}
The Distance matcher is meant to provide corrections for typos. It calculates the edit distance in between the search and the loaded identifiers.
{
"matcher": "distance",
"params": {
"search": "dilterM",
"maximumDistance": 3
}
}
Completion options allow to configure the number of returned completion results.
If specified limits the number of completion results, otherwise return all results.
If set to True, groups all reexports of an identifier under the module it
originated from (the original export is also treated as a "reexport"). These
reexports then populate the exportedFrom field in their completion results and
the module field contains the originating module.
Errors at this point are merely Error strings. Newlines are escaped like \n
and should be taken care of by the editor-plugin.