KEMBAR78
Adds experimental support for running TS Server in a web worker by mjbvz · Pull Request #39656 · microsoft/TypeScript · GitHub
Skip to content

Conversation

@mjbvz
Copy link
Contributor

@mjbvz mjbvz commented Jul 18, 2020

This change makes it possible to run a syntax based TS server in a webworker. This is will let serverless versions of VS Code web run the TypeScript extension with minimal changes.

Screen Shot 2020-07-17 at 11 49 06 PM

Overview
I made the changes in server.ts in order to share as much code as possible between the web and node based servers. However adding web server support require some significant rewriting, so the diff on server.ts is difficult to parse. Here's an overview of the changes:

  • Introduce the concept of a Runtime. Valid values are Node and Web.
  • Move calls to require into the functions that use these modules
  • Wrap existing server logic into startNodeServer
  • Introduce web server with startWebServer. This uses a WorkerSession
  • Add a custom version of ts.sys for web
  • Have the worker server start when it is passed an array of arguments in a message

In order to make the server logic more clear, this change also tries to reduce the reliance on closures and better group declarations vs the logic for starting the server.

Next Steps
I'd like someone from the TS team to help get these changes into a shippable state. This will involve:

  • Adddress todo comments
  • Code cleanup
  • Make sure these changes do not regress node servers
  • Determine if we should add a new tsserver.web.js file instead of having the web worker logic all live in tsserver.js

mjbvz added a commit to mjbvz/vscode that referenced this pull request Jul 21, 2020
This enables running the TS Server on web. This currently requires a special version of the TypeScript server (microsoft/TypeScript#39656)

TODO:

- [ ] Fix commented code in `versionProvider` and `spawner` and `typescriptServiceClient`(they import `fs`)
- [ ] Sync with TS team about how tsserver work will be shipped
@mjbvz
Copy link
Contributor Author

mjbvz commented Jul 21, 2020

@RyanCavanaugh Let's sync tomorrow on the next steps for this work. I've done work on the VS Code side to pull in my forked typescript version (microsoft/vscode#102990) for web builds. This unblocks us but is only a short term workaround

My proposal is that during the start of TS 4.1 development, we collaborate on getting this PR merged into master. In order to keep package sizes down, I think we probably want to ship web support in tsserver.js instead of introducing a new tsserver.web.js. Once the PR is merged, VS Code could then pull the web server from master using nightly builds

@mjbvz mjbvz mentioned this pull request Jul 22, 2020
4 tasks
@typescript-bot typescript-bot added the For Uncommitted Bug PR for untriaged, rejected, closed or missing bug label Jul 23, 2020
mjbvz added 5 commits August 31, 2020 16:03
This change makes it possible to run a syntax old TS server in a webworker. This is will let serverless versions of VS Code web run the TypeScript extension with minimal changes.

As the diff on `server.ts` is difficult to parse, here's an overview of the changes:

- Introduce the concept of a `Runtime`. Valid values are `Node` and `Web`.
- Move calls to `require` into the functions that use these modules
- Wrap existing server logic into `startNodeServer`
- Introduce web server with `startWebServer`. This uses a `WorkerSession`
- Add a custom version of `ts.sys` for web
- Have the worker server start when it is passed an array of arguments in a message

In order to make the server logic more clear, this change also tries to reduce the reliance on closures and better group function declarations vs the server spawning logic.

**Next Steps**
I'd like someone from the TS team to help get these changes into a shippable state. This will involve:

- Adddress todo comments
- Code cleanup
- Make sure these changes do not regress node servers
- Determine if we should add a new `tsserver.web.js` file instead of having the web worker logic all live in `tsserver.js`
Use import types instead
@mjbvz mjbvz force-pushed the web-worker-server branch from 90e38eb to 94faf66 Compare August 31, 2020 23:15
@mjbvz
Copy link
Contributor Author

mjbvz commented Sep 3, 2020

Here's how to test these changes in a web build of VS Code:

  • To build VS code's extensions for the web, simply run yarn watch-web in the VS Code codebase

  • Start the web version of VS code with yarn web.

    Chrome or modern Edge works best for testing

  • VS Code picks up the web version of tsserver from extensions/typescript-language-features/dist/browser/typescript-web

    By default, VS Code pulls in a prebuilt version of TypeScript using an npm install for git://github.com/mjbvz/ts-server-web-build

  • To use your local build instead, build your local copy of TypeScript and then replace the tsserver.web.js file under dist/browser/typescript-web with your built one.

    You can also use a symbolic link so you can iterate more quickly on the TypeScript changes. On MacOS, the command to create this link this would be something like:

    ln -s ~/projects/typescript/built/local/tsserver.js \
        extensions/typescript-language-features/dist/browser/typescript-web/tsserver.web.js
  • To make sure your linked TS version does not get replaced automatically on rebuild, comment out the two CopyPlugin parts of this webpack config

@sheetalkamat

@sheetalkamat
Copy link
Member

Looked at the changes and have updated them:
The things that we need to consider is how to find newLine, useCaseSensitive, readFile (will be needed for lib files in patial semantic mode.. I have gone through rest of the sys methods and have updated them/added comment accordingly. Have set things to some default for now and added TODOs
https://github.com/microsoft/TypeScript/pull/39656/files#diff-7703eaf5820d0a2af1ac3f7dde491b0aR54

newLine: "\r\n", // TODO::
            useCaseSensitiveFileNames: false, // TODO::
            readFile: returnUndefined, // TODO:: read lib Files

            write: postMessage,
            watchFile: returnNoopFileWatcher,
            watchDirectory: returnNoopFileWatcher,

            getExecutingFilePath: returnEmptyString, // TODO:: Locale, lib locations, typesmap location, plugins
            getCurrentDirectory: returnEmptyString, // For inferred project root if projectRoot path is not set, normalizing the paths

This implements enough of `ServerHost` that we can load the standard d.ts files using synchronous XMLHttpRequests.

I also had to patch some code in `editorServices`. I don't know if these changes are correct and need someone on the TS team to review
@mjbvz
Copy link
Contributor Author

mjbvz commented Sep 25, 2020

@sheetalkamat Just pushed 6202c51 that enables support for loading the standard d.ts files that TypeScript ships with. I had to make some changes in editorServices.ts that I am not sure are correct.

The problem is that the standard library files have filenames such as http://localhost:8080/static/extensions/typescript-language-features/dist/browser/typescript-web/lib.dom.d.ts which were being ignored by the host. I don't know if the logic in editorServices needs to be updated or if we try changing the resolved file paths for the standard d.ts files

sheetalkamat and others added 5 commits September 28, 2020 12:09
Co-authored-by: Sheetal Nandi <shkamat@microsoft.com>
This is required for cases where `self.location` does not point to the directory where all the typings are stored
@mjbvz
Copy link
Contributor Author

mjbvz commented Sep 30, 2020

@sheetalkamat Just pushed changes that enables TS Server logging on web. You'll also need to use the latest VS Code build from our main branch to use this

Right now the logger prints to the console. Here's what that looks like:

Screen Shot 2020-09-29 at 6 00 48 PM

As you can see, not ideal for grouped messages but it gets the job done. Let me know if we should look into logging into a VS Code output window instead. For that, we'd likely want to post special logging messages back to VS Code that it can append onto the output

// TODO: is this really needed or can vscode take care of this
How do we handle when opening lib.d.ts as response to goto def in open files
meganrogge pushed a commit to microsoft/vscode that referenced this pull request Nov 18, 2020
@sheetalkamat sheetalkamat marked this pull request as ready for review November 21, 2020 00:20
@typescript-bot
Copy link
Collaborator

This PR doesn't have any linked issues. Please open an issue that references this PR. From there we can discuss and prioritise.

Copy link
Member

@amcasey amcasey left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

First round of comments.

Also, does this create a new output file? Should it?

@sheetalkamat
Copy link
Member

Also, does this create a new output file? Should it?

My understanding is that the goal is to have tsserver.js support webWorker. The delta is not huge (15K) so seems like good option:

12/01/2020  06:26 PM         9,661,335 tsserver.js     <<< Built from master branch
12/01/2020  06:24 PM         9,677,119 tsserver.web.js << Built from this fork

Will let @RyanCavanaugh @mjbvz @DanielRosenwasser @minestarks chime in their thoughts.

@sheetalkamat sheetalkamat requested a review from amcasey December 2, 2020 02:32
Copy link
Member

@amcasey amcasey left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll probably add some more organizational notes, but this approach looks fine.

unknownServerMode?: string;
startSession: (option: StartSessionOptions, logger: Logger, cancellationToken: ServerCancellationToken) => void;
}
function start({ args, logger, cancellationToken, serverMode, unknownServerMode, startSession: startServer }: StartInput, platform: string) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems like a convoluted way to share this initialization code. For example, serverMode is extracted from the StartInput and then passed back into startServer. It might be tidier to extract some initialization helpers (e.g. logHeader) and then have each code path (i.e. web and node) call them separately. Such an exercise might also reveal bits that shouldn't be shared (e.g. why does the web server need to print out the node version in its header?).

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.

Labels

Experiment A fork with an experimental idea which might not make it into master For Uncommitted Bug PR for untriaged, rejected, closed or missing bug

Projects

None yet

Development

Successfully merging this pull request may close these issues.

5 participants