(Originally published on deno.com/blog.)
Deno’s mission is to dramatically simplify software development. In Deno 1.37, we’re happy to extend this to interactive Jupyter notebooks. Starting in 1.37, you can use the new deno jupyter
command to create a Deno kernel that can be used within notebooks. Additionally, Deno 1.37 comes with stronger Visual Studio Code and LSP support, better testing performance, improved Node compatibility, and many bug fixes.
If you already have Deno installed, upgrade to version 1.37 in your terminal with:
deno upgrade
If you don’t yet have Deno installed, you can install it with one of the following commands, or many other ways.
MacOS / Linux Install
curl -fsSL https://deno.land/x/install/install.sh | sh
Windows Install
irm https://deno.land/install.ps1 | iex
Here’s an overview of what’s new in Deno 1.37.
Jupyter notebook integration
Deno v1.37 ships with a built-in support for Jupyter notebook kernel, which brings modern JavaScript and TypeScript and unlocks a whole new set of data science and machine learning possibilities.
If you haven’t already, start by installing Jupyter — this command assumes Python and pip are installed on your system.
pip install jupyterlab
If you have Python 3 installed, you might need to use the pip3
command instead. Other install options are covered here.
To get started using Deno in Jupyter, run:
deno jupyter --unstable
and then follow the prompts to complete the installation. More documentation on deno jupyter --unstable
can be found here.
You can now create interactive REPL sessions using either Jupyter Lab or your favorite IDE which supports these notebooks.
Using the Deno kernel in a Jupyter notebook
Using the Deno kernel in a CLI
You have access to all of Deno’s APIs, as well as npm modules right from your notebook.
Not only can you use modern JavaScript in Jupyter notebooks, but you can also import D3 from npm to visualize your data:
Screenshot contributed by Elijah Meeks
In addition, you can also provide rich output from your cells by returning an object that contains the Symbol.for("Jupyter.display")
function:
You can even connect to your hosted Deno KV with Deno KV Connect and pull live data straight into a notebook:
Finally, if you’re using Noteable.io, you can use Deno in your hosted Jupyter notebooks today. Noteable.io’s Chief Architect and a core maintainer of the Jupyter project, Kyle Kelley, describes the possibilities unlocked from this integration:
Deno’s new Jupyter kernel unlocks tremendous potential for data science workflows by people who can write JavaScript. I’m thrilled by the possibilities enabled by native access to Deno’s modern JavaScript runtime and built-in TypeScript support. JavaScript developers can create notebooks to analyze data, build models, and create interactive reports.
Gone are the days of configuring Babel and TypeScript with a Jupyter JavaScript kernel. Since Deno supports ESM imports based on URLs, the Deno kernel makes it incredibly easy to share notebooks that run anywhere. Dependencies are declarative. Organizations can take advantage of existing Jupyter deployments and immediately get access to Deno’s secure runtime sandbox without giving up the flexible workflow Jupyter enables.
For more details on using Deno’s Jupyter integration, check out our docs. We’re looking for feedback from the community on this feature, so please try it out and let us know if you find any problems or missing features.
VSCode extension and language server
Enable via detection of deno.json
The Deno extension is disabled by default for the convenience of users working on non-Deno TypeScript projects. Previously, you must have specified "deno.enable": true
in your editor’s workspace settings to enable it. Now, as long as your project root contains a deno.json
file, you may leave it unspecified. You can still specify "deno.enable": false
to override this.
"deno.disablePaths"
To assist with cases like the above, a new "deno.disablePaths"
setting has been introduced. This is a complement to the existing "deno.enablePaths"
setting. If you have nested non-Deno sub-projects, it will disable the Deno extension for the specified paths.
// .vscode/settings.json
{
"deno.disablePaths": ["./node_api/", "./vsc_extension/"]
}
Update imports on file rename automatically
This is a widely used refactor among many languages. Deno’s language server now offers this functionality — upon renaming a JS/TS file, you will be prompted to optionally update import specifiers discovered across your project which point to that file.
NPM specifier completions
When typing an npm:
URL in an import specifier, you will now be shown autocomplete options for NPM packages. These are retrieved by searching the NPM registry.
import "npm:prea";
// "npm:preact"; ✔️
// "npm:preact-render-to-string";
// "npm:preact-compat";
// "npm:preact-jsx-chai";
// "npm:preact-css-transition-group";
// ...
But potentially more useful: If you’ve typed npm:<package_name>@
, you’ll see autocomplete options for the latest published versions of that package. Looking up valid versions was a tedious part of using NPM specifiers. Now this is done for you.
import "npm:preact@";
// "npm:preact@11.0.0-experimental.1";
// "npm:preact@11.0.0-experimental.0";
// "npm:preact@10.17.1"; ✔️
// "npm:preact@10.17.0";
// "npm:preact@10.16.0";
// ...
Function call completions
Set "deno.suggest.completeFunctionCalls": true
to include parentheses and argument placeholders when selecting an autocomplete option for a function whose signature is known. This is a re-implementation of "typescript.suggest.completeFunctionCalls"
and "javascript.suggest.completeFunctionCalls"
.
// Type the following:
addEventLi
// and select the autocomplete option for `addEventListener`.
// `"deno.suggest.completeFunctionCalls": false`:
addEventListener
// `"deno.suggest.completeFunctionCalls": true`:
addEventListener(type, listener)
deno.cacheOnSave
A new "deno.cacheOnSave"
setting has been introduced. When enabled, the extension will automatically cache new dependencies when you save a file. This is a huge convenience for users who are frequently adding new dependencies to their projects and removes the need to run “Cache dependencies” command from the command pallete. Currently this setting is off by default, but we plan to enable it by default in a future release.
Test explorer and code lens fixes
Many testing API bugs have been fixed since 1.36.0:
- The test explorer no longer freezes when executing tests with steps.
- Test steps are properly replaced in the explorer when their names are changed. Previously a new step would be registered for each character typed, and you had to reset the language server to clean it up.
- In some cases, nested test steps with the same name would not be registered uniquely from one another and share the same entry in the explorer. The ID allocation is fixed now.
- Running all tests via the test explorer no longer evaluates modules which don’t contain any tests.
- Modules without tests are no longer shown in the test explorer.
- Running a single test via code lens, while another test in the file has
{ only: true }
, would previously not run any tests and log a failure because theonly
option was used. Now it behaves as expected and properly ignores other tests. - Running a test named
foo
via the code lens no longer runs other tests in the same file whose names containfoo
(eg.foobar
). Thanks @Leokuma! - The set of files that are included in the test explorer and show the
Run test
code lens are now filtered based ondeno.json
fieldsexclude
,test.exclude
andtest.include
.
Other fixes
- Show source module in the label description for auto-import completion entries.
- Relative and bare specifiers are prioritized over remote URLs for auto-import completion entries.
- Formatting preferences are respected when inserting auto-import code.
- JSON module specifiers which don’t end in
.json
no longer show an erroneous diagnostic about not having a default export. This affects data URLs and specifiers containing a query string or fragment. - Quickfix actions are sorted by
tsc
,deno
, thendeno-lint
. - Import completions now include local JSON modules.
- Match
"deno.enablePaths"
entries by whole path components. Previously file names that were string-prefixed by an entry would be enabled (eg.foo
would enable bothfoo.ts
andfoobar.ts
). Now it must be path-prefixed, i.e. an entry must be the exact name of the file or one of its parent directories. - Always use
--inspect-wait
over--inspect-brk
when debugging a test. There were still cases where the less suitable--inspect-brk
was used. Thanks @jeiea! - Correct handling of language server clients which don’t support
workspace/configuration
requests. These requests were sent to clients incapable of handling them in some cases. - Remove dead
"deno.testing.enable"
setting. This didn’t do anything.
Testing improvements
Continuing improvements to Deno’s testing APIs from the last release, we have some exciting updates this month as well.
Speed improvements
We improved performance of running tests and Deno can now run up to 14,000 tests per second. This was achieved by optimizing the sanitizers that Deno’s test runner has built-in, lowering baseline overhead from 3ms to less that 0.1ms per test case.
We have further plans to improve testing performance and you can expect more speed ups in the next release.
TAP
test reporter
deno test
can now output test results in the Test Anything Protocol. You can enable it by using --reporter=tap
flag:
$ deno test --reporter=tap ./test.ts
TAP version 14
# ./test.ts
ok 1 - test 0
ok 2 - test 1
ok 3 - test 2
ok 4 - test 3
ok 5 - test 4
ok 6 - test 5
ok 7 - test 6
ok 8 - test 7
ok 9 - test 8
ok 10 - test 9
1..10
Thank you to Valentin Anger for implementing this feature.
Shorthand methods for focusing and ignoring tests
You can now focus and ignore tests using shorthand methods Deno.test.only
and Deno.test.ignore
, like so:
// Only run this test
Deno.test.only(function myTest() {
// ...
});
// Do not run this test
Deno.test.ignore(function myTest() {
// ...
});
It’s a little change, but makes it easier and faster to iterate than having to specify an option bag like Deno.test({ only : true }, function myTest() ...
.
Quality of life improvements
Besides changes described above, we have plethora of other small changes that makes testing experience in Deno better:
Filtered suites are not displayed
If you use --filter <name>
flag to run only a subset of tests, the test runner will now only show the filtered tests in the output. This makes it easier to focus on the tests you care about.
Before Deno v1.37:
In Deno v1.37:
ASCII escape characters no longer mangle test names
Characters like \n
, \t
, etc are now properly escaped in test names, so you if you really need to include them in the test name, the reporter output won’t get broken.
--trace-ops
shows more details
If you have leaking ops in your tests, --trace-ops
flag can help you catch where the leak is coming from. With this release, this flag got smarter and can show useful information in more cases.
Filtering by name takes precedence over only
option
If you have a test suite with the only
option set to true
, but you want to run some other test filtered by a name (using --filter <name>
flag), the test will be properly run.
JUnit reporter includes more information
File, line and column attributes are properly added to the produced report.
Exclude internal code from coverage report
The coverage just got faster and less noisy, by excluding internal Deno code from the generated report.
If you have other ideas how to make Deno’s test runner better, please let us know by opening an issue in Deno’s bug tracker.
Node.js compatibility improvements
A bunch of very important fixes to Node.js APIs landed in this release:
child_process
no longer throws if env var isundefined
ornull
crypto
module implements AES GCM cipherhttp
now correctly handlesContent-Length
header, emits"error"
event, destroying request properly cleans up allocated resourceshttp2
adds support forClientHttp2Session
andconnect
APIprocess.title
no longer throwsrepl._builtinLibs
is now supportedtls
now implementsTLSSocket._start
methodworker_threads.Worker
now correctly handlesprocess.argv
zlib
supportsdictionary
optionrequire
now uses canonicalized paths for loading content- Additonal npm modules tested and working:
mssql
,mineflayer
,infiscal
, andweb-push
If you want to know more about Node.js compatibility in Deno, check out our Node.js compatibility list.
Quality of life improvements
Vendor as cache override (unstable)
It is now possible to vendor dependencies by simply adding the following to your deno.json
:
{
"vendor": true
}
The next time you run or cache (ex. deno cache mod.ts
), Deno will populate a ./vendor
folder with the remote https/http dependencies used by the program.
Alternatively, vendoring is possible via the --vendor
flag.
Note that Deno treats the ./vendor
folder differently because it is not always possible to map urls directly to a file system.
You will probably find this feature much easier to use than the deno vendor
subcommand as it takes very little overhead to manage and works in more scenarios. For that reason, it might make sense for the deno vendor
subcommand to be removed from the CLI and distributed as a separate tool instead (for more details, see issue #20584).
Import attributes
This release adds support for the Import Attributes proposal. The proposal was earlier know as “import assertions”, and Deno had support for it since version 1.17.
Import attributes allow you to import JSON files using import
declarations:
import jsonData from "./data.json" with { type: "json" };
It also works with dynamic imports:
const jsonData = await import("./data.json", { with: { type: "json" } });
The old syntax using assert
is still supported, but deprecated. We are going to phase it out over the next few releases; in the next releases Deno will print a warning if you use the deprecated assert
syntax in your own code.
Deno.Server.prototype.shutdown()
This release brings a new, unstable API: Deno.Server.prototype.shutdown(): Promise<void>
.
// Start a server...
const server = Deno.serve((_req) => new Response("hello world!"));
// ...and close it gracefully after 3 seconds.
setTimeout(async () => {
await server.shutdown();
}, 3000);
Before this release, it was only possible to close the server “abruptly”, by using the AbortSignal
passed to the server during construction. However it’s not always desirable to close the server in such fashion, in most cases you want to wait for all the in-flight requests to finish before closing down the server.
This new API is unstable, because we want to gather feedback from the community before stabilizing it; additionally we want to add a timeout
parameter, that will control how long the server waits for the pending requests to complete before closing them forcefully.
Performance improvements
This release we optimized numerous Web APIs as well HTTP related APIs (both in Deno
APIs as well as built-in Node.js APIs). Here’s the full list:
Event
andaddEventListener
have undergone a deep optimization, making them much faster;Event.timeStamp
is now always set to 0Headers
was optimized for iteration and lookup scenarios, additionally validation of header names was optimizednode:http
got optimizations for handling headers -IncomingMessageForServer.headers
now uses caching to not compute headers on every accessnode:buffer
was rewritten from ground up to provide faster result when outputting stringsnode:net
received an optimization for socket reads, which makesnpm:ws
package much more performant
All these optimizations provide up to 10% more RPS in HTTP benchmarks.
Thank you to Marcos Casagrande for implementing a lot of these optimizations.
Lockfile v3
The Lockfile format has been updated to version 3. Deno 1.37 will automatically migrate your existing lockfile to the new format on first run, but loading v3 in older versions of Deno will error.
The new format includes information on redirects, which is used by Deno to lock http(s) redirects to the destination they were resolved to at the time they were inserted into the lockfile.
std/url
In this release, std/url
module has been added to the Deno Standard Library. std/url
supports various path-like operations on URLs, which are not supported by the Web standard URL methods or properties.
Currently std/url
has the below 5 methods:
basename
dirname
extname
join
normalize
import {
basename,
dirname,
extname,
join,
normalize,
} from "https://deno.land/std@0.202.0/std/url/mod.ts";
const url = new URL("https://example.com/home/page.html?foo=bar");
basename(url); // => "page.html"
dirname(url); // => new URL("https://example.com/home")
extname(url); // => ".html"
join("https://example.com", "foo", "bar.html"); // => new URL("https://example.com/foo/bar.html")
normalize("https://example.com///about///page.html"); // => new URL("https://example.com/about/page.html")
Thank you to Aritra Karak for implementing this module.
std/ulid
std/ulid
has been added to the Deno Standard Library in this release. std/ulid
supports generating ULID
and decoding the timestamp from the given ULID.
import { decodeTime, ulid } from "https://deno.land/std@0.202.0/ulid/mod.ts";
const id = ulid(); // => "01HARJZ15RFMBVZ1GQ8J4VB6C4"
const timestamp = decode(id); // => 1695189796023
ULID is alternative to UUID, and because the leading part of a ULID is generated from a timestamp, these IDs are lexically sorted in generation time order. This property is convenient when you use it for a part of keys for Deno KV items.
Thank you to Asher Gomez for suggesting this feature and thank you to Lino Le Van for implementing it.
V8 11.8 and TypeScript 5.2.2
Finally, Deno v1.37 ships with V8 11.8 and TypeScript 5.2.2.
Still want to learn more?
Believe it or not, the changes listed above still don’t tell you everything that got better in 1.37. You can view the full list of pull requests merged in Deno 1.37 on GitHub here.
Thank you to our community contributors!
We couldn’t build Deno without the help of our community! Whether by answering questions in our community Discord server or reporting bugs, we are incredibly grateful for your support. In particular, we’d like to thank the following people for their contributions to Deno 1.37: Adam Powers, Alexander Michaud, Curran McConnell, Evan, Fabian, Filip Skokan, Jakub Jirutka, Jonathan Rezende, Juan Gonzalez, Kira, Kyle Kelley, Laurence Rowe, Leigh McCulloch, Marcos Casagrande, Shreyas, Valentin Anger, VlkrS, await-ovo, lionel-rowe, osddeitf, sigmaSd, zuisong, 林炳权, 第二扩展.
Would you like to join the ranks of Deno community contributors? Check out our contribution docs here, and we’ll see you on the list next time.
Thank you for catching up with our 1.37 release, and we hope you love building with Deno!
🍋 Did you know? Fresh got even fresher.
Deno Fresh v1.4 released a few weeks ago, featuring faster page loads with an optional AOT compile step, async layouts, route groups, and more. Learn more about the latest release of the modern web framework for Deno.