Skip to content

Latest commit

 

History

History
472 lines (346 loc) · 20.3 KB

vscode.md

File metadata and controls

472 lines (346 loc) · 20.3 KB
id title sidebar_label
vscode
Extension for Visual Studio Code
Visual Studio Code

The Sorbet extension for Visual Studio Code integrates with the Sorbet language server to provide IDE-like features for typed Ruby files.

Installing and enabling the Sorbet extension

Install the Sorbet extension from the VS Code extension marketplace. Then, add the following configuration to your workspace's settings.json:

"sorbet.enabled": true

The next time you open a Ruby file in the workspace, Sorbet will automatically try to run via the following command:

bundle exec srb typecheck --lsp

If needed, you can customize how the extension launches Sorbet via the sorbet.lspConfigs setting:

"sorbet.lspConfigs": [{
    "id": "stable",
    "name": "Sorbet",
    "description": "Stable Sorbet Ruby IDE features",
    "cwd": "${workspaceFolder}",
    "command": [
      "bundle",
      "exec",
      "srb",
      "typecheck",
      "--lsp"
    ]
}]

Once Sorbet is activated, it will display its status in VS Code's status line. For example, this is what you will see when Sorbet is busy typechecking your latest edits:

For the best experience, Sorbet requires Watchman, which listens for changes to the files on disk in addition to edits that happen to files open in the editor. For example, without Watchman installed, Sorbet will not detect when files have changed on disk due to things like changing the currently checked out branch.

Sorbet simply requires that the watchman binary be somewhere visible on the PATH environment variable. There are installation instructions for various platforms in the Watchman docs.

If you cannot install watchman to somewhere on the PATH, you can use the --watchman-path=... command line flag to srb tc to specify a path to the watchman binary.

If you cannot install watchman at all, pass the --disable-watchman flag to srb tc. This will mean that Sorbet only reads the files from disk at startup, and afterwards only ever sees contents of files that have been opened or changed in the editor.

You can use the sorbet.lspConfigs setting described above to have the VS Code extension always pass these command line flags when starting Sorbet.

Features

Live error squiggles for Sorbet typechecking errors (demo):

Type information and documentation on hover (demo):

Go to definition / type definition (demo):

Find all references (demo):

Autocomplete, including sig suggestion (demo):

Rename constants and methods (demo):

Quick fixes (autocorrects) on errors (demo):

Workspace symbol search:

Switching between configurations

The Sorbet extension supports switching between multiple configurations to make it easy to try out experimental features. By default, it ships with three configurations: stable, beta, and experimental. Workspaces can specify alternative configurations via the sorbet.lspConfigs setting.

Users can select between these configurations on-the-fly by clicking on "Sorbet" in VS Code's status line and selecting "Configure Sorbet". Sorbet will then restart in the chosen configuration. Sorbet will also remember this configuration choice for the user's future sessions in the workspace.

Disabling the Sorbet extension

There are multiple ways to disable the Sorbet extension depending on your goals.

  • You can click on Sorbet in VS Code's status line and click on Disable Sorbet, which will immediately stop Sorbet in that workspace. You will need to click on Enable Sorbet to reenable Sorbet in that workspace. This is a handy way to temporarily disable Sorbet if it is causing problems.
  • Workspaces can set the sorbet.enabled setting to false, which prevents Sorbet from running in the workspace.

Troubleshooting and FAQ

Startup

Error: "Sorbet's language server requires a single input directory. However, 0 are configured"

This error can happen if you have not initialized Sorbet in your project. Please follow the instructions to initialize Sorbet.

If initializing Sorbet in your project is not desirable or possible, an alternative fix is to override the default extension configuration in the project's .vscode/settings.json file and provide the project directory as ".":

"sorbet.lspConfigs": [{
    "id": "stable",
    "name": "Sorbet",
    "description": "Stable Sorbet Ruby IDE features",
    "cwd": "${workspaceFolder}",
    "command": [
      "bundle",
      "exec",
      "srb",
      "typecheck",
      "--lsp",
      "."
    ]
}]

I'm not seeing "Sorbet: Disabled" or "Sorbet: Idle" in the status bar of my VSCode window

That means the extension isn't active. Here are some steps to try:

  • Did you open a Ruby file? The Sorbet extension isn't activated until you open at least one Ruby file in your editor.
  • Make sure your VS Code window is wide enough to display the entire contents of the status bar.
  • Ensure that you are not using VS Code's Multi-root Workspaces feature, which this extension does not support.

Sorbet keeps restarting.

Click on Sorbet in the status bar, and then click on View Output. A log should pop up, and will typically contain some error messages that are causing the restart.

If you see an error that looks like this:

[Error - 9:36:32 AM] Connection to server got closed. Server will not be restarted.
Running Sorbet LSP with:
    bundle exec srb typecheck --lsp
Could not locate Gemfile or .bundle/ directory

...then you probably need to run bundle install to install Sorbet.

Diagnostics / error squiggles

Sorbet is reporting diagnostics (error squiggles) that don't make sense.

If the errors are persistent, and you can reproduce your problem in the sandbox at https://sorbet.run/, then you've found an issue with Sorbet in general, not necessarily the VSCode Sorbet extension. Please file a bug tagged with "IDE" on the issue tracker.

If the errors are not persistent:

  • First, is Sorbet still typechecking? If it is, wait for the status bar to display "Sorbet: Idle." before judging the diagnostics.
  • If the diagnostics remain nonsensical when Sorbet is Idle, try restarting Sorbet (click on Sorbet in the status bar, and click on Restart Sorbet).
  • If the diagnostics become fixed, then you might have discovered a bug in Sorbet.

If you arrive at a set of edits that mess up the diagnostics, please file a bug on the issue tracker.

Hover

When I hover on something, VS Code shows "Loading..."

Is Sorbet still typechecking or initializing? If so, this is expected behavior; Sorbet cannot show you type information until it finishes catching up with your edits. "Loading..." should get replaced with hover information once the status bar displays "Sorbet: Typechecking in background" or "Sorbet: Idle".

Hover doesn't work / hover isn't showing information for my file

Is the file untyped (# typed: false or ignore)? Hover only works in typed files. You will need to make your file # typed: true.

Does the file have a syntax error? If so, you need to resolve it before hover will work again.

If the file is typed, Sorbet is "Idle", and hover isn't working, try to reproduce the problem on https://sorbet.run and file a bug on the issue tracker.

Hover is showing incorrect type information for something.

Does your file have a syntax error? If so, resolve it before proceeding.

If the problem persists with your syntax errors fixed, you may have found a bug in Sorbet! Check to see if it's in the GitHub issue tracker. If you don't see anything relevant there, try to golf the problem down to something small on https://sorbet.run and file a bug on the issue tracker.

Hover is not showing documentation that I've written / is showing incorrect documentation.

Internally, Hover uses the "Go to Definition" logic to locate documentation. If "Go to Definition" takes you to the wrong location, then Sorbet doesn't have the type information it needs to locate your documentation.

Did you include an empty line between your documentation and the thing you are defining? We expect documentation to immediately precede the definition (or its sig).

# valid documentation location
def foo; end

# valid documentation location
sig {void}
def foo; end

sig {void}
# valid documentation location
def foo; end

# invalid documentation location

def foo; end

Is this an instance or class variable? The variable must be defined with T.let, and the documentation must precede the T.let. Otherwise, Sorbet doesn’t see that variable as having ever been defined.

Otherwise, try to reproduce the issue on https://sorbet.run/ and file a bug on the issue tracker.

Go to Definition/Go to Type Definition/Find all References

Go to Definition/Go to Type Definition/Find all References is not working / Find all References is missing some expected results.

First, make sure that Sorbet is running. You should see "Sorbet: Idle" in VS Code's status bar.

It's possible that the feature is working as intended. Go to Definition and Find all References are not available in all circumstances. A ‘Yes’ entry in the following table indicates two distinct things:

  • A language construct where these features work. For example, you can right click a class reference and go to its definition or find its references in typed and untyped files.
  • An item that will be included in the results of these features. For example, the result of Find all References on a method includes method calls in typed files, but not method calls in untyped files.
# typed: true or above # typed: false
Class/module/constant definition or reference Yes Yes
ivar (@foo) Yes, if defined* Yes, if defined*
cvar (@@foo) Yes, if defined* Yes, if defined*
Method definition Yes Results only**
Method call Yes No
Local variable Yes No

* Indicates that the variable must be defined with T.let. Otherwise, Sorbet doesn’t see that variable as having ever been defined.

** You cannot use either feature from this language construct, but this item will be included in Go to Definition/Find all References results from typed files.

We have these restrictions in place to avoid weird/nonsensical behavior caused by untyped files, which may have partial and potentially incorrect type information. We heartily encourage you to type files to gain access to these features.

Note that some items marked "Yes" in the table may not work with these features if Sorbet does not have the necessary type information. In particular, method calls on an object foo where foo is untyped will not be included in Find all References and will not work with Go to Definition because Sorbet is unable to resolve which method is being called at that location. Using # typed: strict should suss out most of these untyped locations on a per-file basis.

Find all References is slow.

The speed of Find all References depends on how many files contain an identifier with the same name, as we use that information as a first-pass filter over the source files before refining to actual references. For example, searching for references to a class's initialize will involve a scan over most files in a project even if the specific initialize you are looking for is only used in one file.

Find all References also waits for "Typechecking in background..." to complete so that it does not contend with typechecking for CPU time.

Go to Definition/Go to Type Definition brought me to what I believe is the wrong location.

Ensure that you see "Sorbet: Idle" and not "Sorbet: Disabled" at the bottom of VS Code. If Sorbet is enabled and it is returning a weird/unexpected definition site, please try to reproduce the issue on https://sorbet.run/ and file a bug on the issue tracker.

Go to Definition/Go to Type Definition/Find all References brought me to a file that I cannot edit.

These features may return results in type definitions for core Ruby libraries, which are baked directly into the Sorbet executable and are not present on the file system.

In order to display these files in your editor and to support navigating through them, we've configured the Sorbet extension to display them in this read-only view. Note that certain extension features, like hover and Go to Definition, will not function in some of these special files.

Completion

I don't see any completion results.

  • Are you in a typed: false file? No completion results are expected.
  • Is the place where you're trying to see results unreachable? For example, after a return statement, or in an else condition that can't happen? Sorbet can't provide completion results here.
  • Can you see completion results for other things? Sorbet only supports completing local variables, methods, keywords, suggested sigs, classes, modules, and constants right now. Notably, it doesn't support completing the names of instance variables.

I don't see any completion results right after I type A:: or x.

You'll have to type at least one character after the dot (like x.f) or after the colon (like A::B) before completion results show up.

We tried to get this working before the initial ship, but it ended up being a more complicated change than we expected. We have a couple ideas how to support this, so expect this to be supported in the future.

The completion results look wrong.

Completion results can come from many different extensions, not just Sorbet. You can try to figure out what extension returned the results by looking at the icon that VS Code shows in the completion list:

Results from Sorbet will only ever have 1 of 6 icons (currently): method, variable, field, class, interface, module, enum, keyword, and snippet.

Notably, the abc icon (word) means the results came either from VS Code’s editor.wordBasedSuggestions setting or some other generic autocomplete extension.

Also, snippet results can come from other extensions. Snippet results that come from Sorbet will always say (sorbet) somewhere in the snippet description. Sorbet does not have control over any snippet results that don't say (sorbet) in them; if they look wrong, the only suggestion is to turn them off.

Can I have Sorbet only suggest method names, not the entire snippet, with types?

Sorbet inserts a suggested snippet into the document when accepting a completion result.

  • Snippet results will have highlighted sections inside them.
  • These represent "holes" (tabstops) that you'll need to fill in—the aim is that every tabstop is for a required argument (i.e., optional / default arguments won't be present).
  • As the default text for each of these holes, Sorbet uses the type of the corresponding argument.
  • Press TAB to cycle through the holes (tabstops), or press ESC to deselect all the tabstops.

It is not possible to opt-out of these completion snippets. If you find that this is annoying, please let us know.

Reporting metrics

Sorbet does not require metrics gathering for full functionality. If you are seeing a log line like "Sorbet metrics gathering disabled," Sorbet is working as intended.

It is possible to ask the Sorbet VS Code extension to collect and report usage metrics. This is predominantly useful if you maintain a large Ruby codebase that uses Sorbet, and want to gather metrics on things like daily users and editor responsiveness.

To start gathering metrics, implement a custom VS Code command using the name sorbet.metrics.getExportedApi.

The implementation of this command should simply return an object like this:

{
  metricsEmitter: ...
}

The metricsEmitter value should conform to the MetricsEmitter interface declared in the Sorbet VS Code extension source code. Most likely, you will want to implement this interface by importing a StatsD client, connecting to an internal metrics reporting host, and forwarding requests from the MetricEmitter interface function calls to the StatsD client of your choice.