2019-02-11 17:41:13 -05:00
|
|
|
# Deno Manual
|
|
|
|
|
|
|
|
[toc]
|
|
|
|
|
|
|
|
## Disclaimer
|
|
|
|
|
|
|
|
A word of caution: Deno is very much under development. We encourage brave early
|
|
|
|
adopters, but expect bugs large and small. The API is subject to change without
|
2019-02-12 23:54:08 -05:00
|
|
|
notice. [Bug reports](https://github.com/denoland/deno/issues) do help!
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
## Introduction
|
|
|
|
|
2019-03-05 00:01:32 -05:00
|
|
|
A secure JavaScript/TypeScript runtime built with V8, Rust, and Tokio
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
### Philosophy
|
|
|
|
|
2019-02-12 23:54:08 -05:00
|
|
|
Deno aims to be a productive and secure scripting environment for the modern
|
|
|
|
programmer.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
It will always be distributed as a single executable - and that executable will
|
|
|
|
be sufficient software to run any deno program. Given a URL to a deno program,
|
2019-02-12 23:54:08 -05:00
|
|
|
you should be able to execute it with nothing more than the 50 megabyte deno
|
|
|
|
executable.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
Deno explicitly takes on the role of both runtime and package manager. It uses a
|
|
|
|
standard browser-compatible protocol for loading modules: URLs.
|
|
|
|
|
|
|
|
Deno provides security guarantees about how programs can access your system with
|
|
|
|
the default being the most restrictive secure sandbox.
|
|
|
|
|
|
|
|
Deno provides <a href="https://github.com/denoland/deno_std">a set of reviewed
|
|
|
|
(audited) standard modules</a> that are guaranteed to work with Deno.
|
|
|
|
|
2019-02-12 23:54:08 -05:00
|
|
|
### Goals
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
- Support TypeScript out of the box.
|
|
|
|
|
|
|
|
- Like the browser, allows imports from URLs:
|
|
|
|
|
|
|
|
```typescript
|
2019-03-06 07:23:47 -08:00
|
|
|
import * as log from "https://deno.land/std/log/mod.ts";
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
- Remote code is fetched and cached on first execution, and never updated until
|
|
|
|
the code is run with the `--reload` flag. (So, this will still work on an
|
|
|
|
airplane. See `~/.deno/src` for details on the cache.)
|
|
|
|
|
|
|
|
- Uses "ES Modules" and does not support `require()`.
|
|
|
|
|
|
|
|
- File system and network access can be controlled in order to run sandboxed
|
|
|
|
code. Access between V8 (unprivileged) and Rust (privileged) is only done via
|
|
|
|
serialized messages defined in this
|
|
|
|
[flatbuffer](https://github.com/denoland/deno/blob/master/src/msg.fbs). This
|
|
|
|
makes it easy to audit. For example, to enable write access use the flag
|
|
|
|
`--allow-write` or for network access `--allow-net`.
|
|
|
|
|
|
|
|
- Only ship a single executable.
|
|
|
|
|
|
|
|
- Always dies on uncaught errors.
|
|
|
|
|
2019-02-12 23:54:08 -05:00
|
|
|
- Browser compatible: The subset of Deno programs which are written completely
|
|
|
|
in JavaScript and do not use the global `Deno` namespace (or feature test for
|
|
|
|
it), ought to also be able to be run in a modern web browser without change.
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
- [Aims to support top-level `await`.](https://github.com/denoland/deno/issues/471)
|
|
|
|
|
2019-03-05 00:01:32 -05:00
|
|
|
- Be able to serve HTTP efficently.
|
|
|
|
([Currently it is relatively slow.](https://deno.land/benchmarks.html#req-per-sec))
|
|
|
|
|
|
|
|
- Provide useful tooling out of the box: Built-in command-line debugger
|
|
|
|
[not yet](https://github.com/denoland/deno/issues/1120), built-in lint
|
|
|
|
[not yet](https://github.com/denoland/deno/issues/1880), dependency inspector
|
|
|
|
(`deno --info`), built-in code formatter (`deno --fmt`),
|
|
|
|
|
2019-02-12 23:54:08 -05:00
|
|
|
### Non-goals
|
|
|
|
|
|
|
|
- No `package.json`.
|
|
|
|
|
|
|
|
- No npm.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
2019-02-12 23:54:08 -05:00
|
|
|
- Not explicitly compatible with Node.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
## Setup
|
|
|
|
|
|
|
|
### Binary Install
|
|
|
|
|
|
|
|
Deno works on OSX, Linux, and Windows. Deno is a single binary executable. It
|
|
|
|
has no external dependencies.
|
|
|
|
|
|
|
|
[deno_install](https://github.com/denoland/deno_install) provides convenience
|
|
|
|
scripts to download and install the binary.
|
|
|
|
|
|
|
|
Using Shell:
|
|
|
|
|
2019-02-19 00:19:43 +01:00
|
|
|
```sh
|
|
|
|
curl -fsSL https://deno.land/x/install/install.sh | sh
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
Or using PowerShell:
|
|
|
|
|
|
|
|
```powershell
|
2019-02-19 00:19:43 +01:00
|
|
|
iwr https://deno.land/x/install/install.ps1 | iex
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
Deno can also be installed manually, by downloading a tarball or zip file at
|
|
|
|
[github.com/denoland/deno/releases](https://github.com/denoland/deno/releases).
|
|
|
|
These packages contain just a single executable file. You will have to set the
|
|
|
|
executable bit on Mac and Linux.
|
|
|
|
|
|
|
|
Once it's installed and in your \$PATH, try it:
|
|
|
|
|
|
|
|
```
|
2019-02-19 01:18:00 +02:00
|
|
|
deno https://deno.land/welcome.ts
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
### Build from source
|
|
|
|
|
2019-02-16 00:15:44 +08:00
|
|
|
```bash
|
2019-02-11 17:41:13 -05:00
|
|
|
# Fetch deps.
|
|
|
|
git clone --recurse-submodules https://github.com/denoland/deno.git
|
|
|
|
cd deno
|
|
|
|
./tools/setup.py
|
|
|
|
|
|
|
|
# You may need to ensure that sccache is running.
|
|
|
|
# (TODO it's unclear if this is necessary or not.)
|
|
|
|
# prebuilt/mac/sccache --start-server
|
|
|
|
|
|
|
|
# Build.
|
|
|
|
./tools/build.py
|
|
|
|
|
|
|
|
# Run.
|
|
|
|
./target/debug/deno tests/002_hello.ts
|
|
|
|
|
|
|
|
# Test.
|
|
|
|
./tools/test.py
|
|
|
|
|
|
|
|
# Format code.
|
2019-02-12 15:20:32 -05:00
|
|
|
./tools/format.py
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
#### Prerequisites
|
|
|
|
|
|
|
|
To ensure reproducible builds, deno has most of its dependencies in a git
|
|
|
|
submodule. However, you need to install separately:
|
|
|
|
|
|
|
|
1. [Rust](https://www.rust-lang.org/en-US/install.html) >= 1.31.1
|
|
|
|
2. [Node](https://nodejs.org/)
|
|
|
|
3. Python 2.
|
|
|
|
[Not 3](https://github.com/denoland/deno/issues/464#issuecomment-411795578).
|
|
|
|
|
|
|
|
Extra steps for Mac users:
|
|
|
|
|
|
|
|
1. [XCode](https://developer.apple.com/xcode/)
|
|
|
|
2. Openssl 1.1: `brew install openssl@1.1` (TODO: shouldn't be necessary)
|
|
|
|
|
|
|
|
Extra steps for Windows users:
|
|
|
|
|
|
|
|
1. Add `python.exe` to `PATH` (e.g. `set PATH=%PATH%;C:\Python27\python.exe`)
|
|
|
|
2. Get [VS Community 2017](https://www.visualstudio.com/downloads/) with
|
|
|
|
`Desktop development with C++` toolkit and make sure to select the following
|
|
|
|
required tools listed below along with all C++ tools.
|
|
|
|
- Windows 10 SDK >= 10.0.17134
|
|
|
|
- Visual C++ ATL for x86 and x64
|
|
|
|
- Visual C++ MFC for x86 and x64
|
|
|
|
- C++ profiling tools
|
|
|
|
3. Enable `Debugging Tools for Windows`. Go to `Control Panel` → `Programs` →
|
|
|
|
`Programs and Features` → Select
|
|
|
|
`Windows Software Development Kit - Windows 10` → `Change` → `Change` → Check
|
|
|
|
`Debugging Tools For Windows` → `Change` -> `Finish`.
|
2019-03-04 20:24:51 -08:00
|
|
|
4. Make sure you are using git version 2.19.2.windows.1 or newer.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
#### Other useful commands
|
|
|
|
|
2019-02-16 00:15:44 +08:00
|
|
|
```bash
|
2019-02-11 17:41:13 -05:00
|
|
|
# Call ninja manually.
|
|
|
|
./third_party/depot_tools/ninja -C target/debug
|
|
|
|
|
|
|
|
# Build a release binary.
|
|
|
|
DENO_BUILD_MODE=release ./tools/build.py :deno
|
|
|
|
|
|
|
|
# List executable targets.
|
|
|
|
./third_party/depot_tools/gn ls target/debug //:* --as=output --type=executable
|
|
|
|
|
|
|
|
# List build configuration.
|
|
|
|
./third_party/depot_tools/gn args target/debug/ --list
|
|
|
|
|
|
|
|
# Edit build configuration.
|
|
|
|
./third_party/depot_tools/gn args target/debug/
|
|
|
|
|
|
|
|
# Describe a target.
|
|
|
|
./third_party/depot_tools/gn desc target/debug/ :deno
|
|
|
|
./third_party/depot_tools/gn help
|
|
|
|
|
|
|
|
# Update third_party modules
|
|
|
|
git submodule update
|
|
|
|
```
|
|
|
|
|
|
|
|
Environment variables: `DENO_BUILD_MODE`, `DENO_BUILD_PATH`, `DENO_BUILD_ARGS`,
|
|
|
|
`DENO_DIR`.
|
|
|
|
|
|
|
|
## API reference
|
|
|
|
|
|
|
|
### deno --types
|
|
|
|
|
|
|
|
To get an exact reference of deno's runtime API, run the following in the
|
|
|
|
command line:
|
|
|
|
|
|
|
|
```
|
|
|
|
deno --types
|
|
|
|
```
|
|
|
|
|
|
|
|
[This is what the output looks like.](https://gist.github.com/ry/46da4724168cdefa763e13207d27ede5)
|
|
|
|
|
|
|
|
### Reference websites
|
|
|
|
|
|
|
|
[TypeScript Deno API](https://deno.land/typedoc/index.html).
|
|
|
|
|
|
|
|
If you are embedding deno in a Rust program, see
|
|
|
|
[Rust Deno API](https://deno.land/rustdoc/deno/index.html).
|
|
|
|
|
|
|
|
## Examples
|
|
|
|
|
|
|
|
### An implementation of the unix "cat" program
|
|
|
|
|
|
|
|
In this program each command-line argument is assumed to be a filename, the file
|
|
|
|
is opened, and printed to stdout.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
(async () => {
|
2019-02-13 02:08:56 +11:00
|
|
|
for (let i = 1; i < Deno.args.length; i++) {
|
|
|
|
let filename = Deno.args[i];
|
|
|
|
let file = await Deno.open(filename);
|
|
|
|
await Deno.copy(Deno.stdout, file);
|
2019-02-11 17:41:13 -05:00
|
|
|
file.close();
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
```
|
|
|
|
|
|
|
|
The `copy()` function here actually makes no more than the necessary kernel ->
|
|
|
|
userspace -> kernel copies. That is, the same memory from which data is read
|
|
|
|
from the file, is written to stdout. This illustrates a general design goal for
|
|
|
|
I/O streams in Deno.
|
|
|
|
|
|
|
|
Try the program:
|
|
|
|
|
|
|
|
```
|
2019-03-06 07:23:47 -08:00
|
|
|
> deno https://deno.land/std/examples/cat.ts /etc/passwd
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
### TCP echo server
|
|
|
|
|
|
|
|
This is an example of a simple server which accepts connections on port 8080,
|
|
|
|
and returns to the client anything it sends.
|
|
|
|
|
|
|
|
```ts
|
2019-02-13 02:08:56 +11:00
|
|
|
const { listen, copy } = Deno;
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
(async () => {
|
|
|
|
const addr = "0.0.0.0:8080";
|
|
|
|
const listener = listen("tcp", addr);
|
|
|
|
console.log("listening on", addr);
|
|
|
|
while (true) {
|
|
|
|
const conn = await listener.accept();
|
|
|
|
copy(conn, conn);
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
```
|
|
|
|
|
|
|
|
When this program is started, the user is prompted for permission to listen on
|
|
|
|
the network:
|
|
|
|
|
|
|
|
```
|
2019-03-06 07:23:47 -08:00
|
|
|
> deno https://deno.land/std/examples/echo_server.ts
|
2019-02-11 17:41:13 -05:00
|
|
|
⚠️ Deno requests network access to "listen". Grant? [yN] y
|
|
|
|
listening on 0.0.0.0:8080
|
|
|
|
```
|
|
|
|
|
|
|
|
For security reasons, deno does not allow programs to access the network without
|
|
|
|
explicit permission. To avoid the console prompt, use a command-line flag:
|
|
|
|
|
|
|
|
```
|
2019-03-06 07:23:47 -08:00
|
|
|
> deno https://deno.land/std/examples/echo_server.ts --allow-net
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
To test it, try sending a HTTP request to it by using curl. The request gets
|
|
|
|
written directly back to the client.
|
|
|
|
|
|
|
|
```
|
|
|
|
> curl http://localhost:8080/
|
|
|
|
GET / HTTP/1.1
|
|
|
|
Host: localhost:8080
|
|
|
|
User-Agent: curl/7.54.0
|
|
|
|
Accept: */*
|
|
|
|
```
|
|
|
|
|
|
|
|
It's worth noting that like the `cat.ts` example, the `copy()` function here
|
|
|
|
also does not make unnecessary memory copies. It receives a packet from the
|
|
|
|
kernel and sends back, without further complexity.
|
|
|
|
|
2019-03-04 17:04:19 +01:00
|
|
|
### Inspecting and revoking permissions
|
|
|
|
|
|
|
|
Sometimes a program may want to revoke previously granted permissions. When a
|
|
|
|
program, at a later stage, needs those permissions, a new prompt will be
|
|
|
|
presented to the user.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
const { permissions, revokePermission, open, remove } = Deno;
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
// lookup a permission
|
|
|
|
if (!permissions().write) {
|
|
|
|
throw new Error("need write permission");
|
|
|
|
}
|
|
|
|
|
|
|
|
const log = await open("request.log", "a+");
|
|
|
|
|
|
|
|
// revoke some permissions
|
|
|
|
revokePermission("read");
|
|
|
|
revokePermission("write");
|
|
|
|
|
|
|
|
// use the log file
|
|
|
|
await log.write(encoder.encode("hello\n"));
|
|
|
|
|
|
|
|
// this will prompt for the write permission or fail.
|
|
|
|
await remove("request.log");
|
|
|
|
})();
|
|
|
|
```
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
### File server
|
|
|
|
|
|
|
|
This one serves a local directory in HTTP.
|
|
|
|
|
|
|
|
```
|
|
|
|
alias file_server="deno --allow-net --allow-read \
|
2019-03-06 07:23:47 -08:00
|
|
|
https://deno.land/std/http/file_server.ts"
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
Run it:
|
|
|
|
|
|
|
|
```
|
|
|
|
% file_server .
|
2019-03-06 07:23:47 -08:00
|
|
|
Downloading https://deno.land/std/http/file_server.ts...
|
2019-02-11 17:41:13 -05:00
|
|
|
[...]
|
|
|
|
HTTP server listening on http://0.0.0.0:4500/
|
|
|
|
```
|
|
|
|
|
|
|
|
And if you ever want to upgrade to the latest published version:
|
|
|
|
|
|
|
|
```
|
|
|
|
file_server --reload
|
|
|
|
```
|
|
|
|
|
2019-02-19 00:52:46 +01:00
|
|
|
### Run subprocess
|
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
[API Reference](https://deno.land/typedoc/index.html#run)
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
Example:
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
```ts
|
|
|
|
async function main() {
|
|
|
|
// create subprocess
|
|
|
|
const p = Deno.run({
|
|
|
|
args: ["echo", "hello"]
|
|
|
|
});
|
|
|
|
|
|
|
|
// await its completion
|
|
|
|
await p.status();
|
|
|
|
}
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
main();
|
2019-02-19 00:52:46 +01:00
|
|
|
```
|
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
Run it:
|
2019-02-19 00:52:46 +01:00
|
|
|
|
|
|
|
```
|
2019-02-21 21:52:35 +01:00
|
|
|
> deno --allow-run ./subprocess_simple.ts
|
|
|
|
hello
|
2019-02-19 00:52:46 +01:00
|
|
|
```
|
|
|
|
|
2019-03-03 06:24:33 +09:00
|
|
|
By default when you use `Deno.run()` subprocess inherits `stdin`, `stdout` and
|
2019-02-19 00:52:46 +01:00
|
|
|
`stdout` of parent process. If you want to communicate with started subprocess
|
|
|
|
you can use `"piped"` option.
|
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
```ts
|
|
|
|
async function main() {
|
|
|
|
const decoder = new TextDecoder();
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
const fileNames = Deno.args.slice(1);
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
const p = Deno.run({
|
|
|
|
args: [
|
|
|
|
"deno",
|
|
|
|
"--allow-read",
|
2019-03-06 07:23:47 -08:00
|
|
|
"https://deno.land/std/examples/cat.ts",
|
2019-02-21 21:52:35 +01:00
|
|
|
...fileNames
|
|
|
|
],
|
|
|
|
stdout: "piped",
|
|
|
|
stderr: "piped"
|
|
|
|
});
|
|
|
|
|
|
|
|
const { code } = await p.status();
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
const rawOutput = await p.output();
|
|
|
|
Deno.stdout.write(rawOutput);
|
2019-02-19 00:52:46 +01:00
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
Deno.exit(code);
|
2019-02-19 00:52:46 +01:00
|
|
|
}
|
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
main();
|
2019-02-19 00:52:46 +01:00
|
|
|
```
|
|
|
|
|
|
|
|
When you run it:
|
|
|
|
|
|
|
|
```
|
2019-02-21 21:52:35 +01:00
|
|
|
> deno ./subprocess.ts --allow-run <somefile>
|
2019-02-19 00:52:46 +01:00
|
|
|
[file content]
|
|
|
|
|
2019-02-21 21:52:35 +01:00
|
|
|
> deno ./subprocess.ts --allow-run non_existent_file.md
|
2019-02-19 00:52:46 +01:00
|
|
|
|
|
|
|
Uncaught NotFound: No such file or directory (os error 2)
|
|
|
|
at DenoError (deno/js/errors.ts:19:5)
|
|
|
|
at maybeError (deno/js/errors.ts:38:12)
|
|
|
|
at handleAsyncMsgFromRust (deno/js/dispatch.ts:27:17)
|
|
|
|
```
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
### Linking to third party code
|
|
|
|
|
|
|
|
In the above examples, we saw that Deno could execute scripts from URLs. Like
|
|
|
|
browser JavaScript, Deno can import libraries directly from URLs. This example
|
|
|
|
uses a URL to import a test runner library:
|
|
|
|
|
|
|
|
```ts
|
2019-03-02 14:58:50 -05:00
|
|
|
import {
|
|
|
|
test,
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals,
|
2019-03-02 14:58:50 -05:00
|
|
|
runIfMain
|
2019-03-06 07:23:47 -08:00
|
|
|
} from "https://deno.land/std/testing/mod.ts";
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
test(function t1() {
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals("hello", "hello");
|
2019-02-11 17:41:13 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
test(function t2() {
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals("world", "world");
|
2019-02-11 17:41:13 -05:00
|
|
|
});
|
2019-03-02 14:58:50 -05:00
|
|
|
|
|
|
|
runIfMain(import.meta);
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
Try running this:
|
|
|
|
|
|
|
|
```
|
2019-03-02 14:58:50 -05:00
|
|
|
> deno test.ts --reload
|
2019-02-11 17:41:13 -05:00
|
|
|
running 2 tests
|
2019-03-02 14:58:50 -05:00
|
|
|
test t1 ... ok
|
|
|
|
test t2 ... ok
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
test result: ok. 2 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out
|
2019-03-02 14:58:50 -05:00
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
Note that we did not have to provide the `--allow-net` flag for this program,
|
|
|
|
and yet it accessed the network. The runtime has special access to download
|
|
|
|
imports and cache them to disk.
|
|
|
|
|
|
|
|
Deno caches remote imports in a special directory specified by the `$DENO_DIR`
|
2019-02-13 14:57:00 +01:00
|
|
|
environmental variable. It defaults to the system's cache directory if
|
|
|
|
`$DENO_DIR` is not specified. The next time you run the program, no downloads
|
|
|
|
will be made. If the program hasn't changed, it won't be recompiled either. The
|
|
|
|
default directory is:
|
|
|
|
|
|
|
|
- On Linux/Redox: `$XDG_CACHE_HOME/deno` or `$HOME/.cache/deno`
|
|
|
|
- On Windows: `%LOCALAPPDATA%/deno` (`%LOCALAPPDATA%` = `FOLDERID_LocalAppData`)
|
|
|
|
- On macOS: `$HOME/Library/Caches/deno`
|
|
|
|
- If something fails, it falls back to `$HOME/.deno`
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
**But what if `https://deno.land/` goes down?** Relying on external servers is
|
|
|
|
convenient for development but brittle in production. Production software should
|
|
|
|
always bundle its dependencies. In Deno this is done by checking the `$DENO_DIR`
|
|
|
|
into your source control system, and specifying that path as the `$DENO_DIR`
|
|
|
|
environmental variable at runtime.
|
|
|
|
|
|
|
|
**How do you import to a specific version?** Simply specify the version in the
|
|
|
|
URL. For example, this URL fully specifies the code being run:
|
|
|
|
`https://unpkg.com/liltest@0.0.5/dist/liltest.js`. Combined with the
|
|
|
|
aforementioned technique of setting `$DENO_DIR` in production to stored code,
|
|
|
|
one can fully specify the exact code being run, and execute the code without
|
|
|
|
network access.
|
|
|
|
|
|
|
|
**It seems unwieldy to import URLs everywhere. What if one of the URLs links to
|
|
|
|
a subtly different version of a library? Isn't it error prone to maintain URLs
|
|
|
|
everywhere in a large project?** The solution is to import and re-export your
|
|
|
|
external libraries in a central `package.ts` file (which serves the same purpose
|
|
|
|
as Node's `package.json` file). For example, let's say you were using the above
|
|
|
|
testing library across a large project. Rather than importing
|
2019-03-06 07:23:47 -08:00
|
|
|
`"https://deno.land/std/testing/mod.ts"` everywhere, you could create a
|
2019-02-11 17:41:13 -05:00
|
|
|
`package.ts` file the exports the third-party code:
|
|
|
|
|
|
|
|
```ts
|
2019-03-06 20:48:46 -05:00
|
|
|
export { test, assertEquals } from "https://deno.land/std/testing/mod.ts";
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
And throughout project one can import from the `package.ts` and avoid having
|
|
|
|
many references to the same URL:
|
|
|
|
|
|
|
|
```ts
|
2019-03-06 20:48:46 -05:00
|
|
|
import { test, assertEquals } from "./package.ts";
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
This design circumvents a plethora of complexity spawned by package management
|
|
|
|
software, centralized code repositories, and superfluous file formats.
|
|
|
|
|
2019-02-12 21:14:02 -05:00
|
|
|
### Testing if current file is the main program
|
|
|
|
|
2019-03-02 21:02:47 +01:00
|
|
|
To test if the current script has been executed as the main input to the program
|
|
|
|
check `import.meta.main`.
|
2019-02-12 21:14:02 -05:00
|
|
|
|
|
|
|
```ts
|
2019-03-02 21:02:47 +01:00
|
|
|
if (import.meta.main) {
|
2019-02-12 21:14:02 -05:00
|
|
|
console.log("main");
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
## Command line interface
|
|
|
|
|
|
|
|
### Flags
|
|
|
|
|
|
|
|
```
|
|
|
|
> deno -h
|
|
|
|
Usage: deno script.ts
|
|
|
|
|
|
|
|
Options:
|
2019-02-27 00:14:10 -05:00
|
|
|
--allow-read Allow file system read access
|
|
|
|
--allow-write Allow file system write access
|
|
|
|
--allow-net Allow network access
|
|
|
|
--allow-env Allow environment access
|
|
|
|
--allow-run Allow running subprocesses
|
|
|
|
-A, --allow-all Allow all permissions
|
|
|
|
--recompile Force recompilation of TypeScript code
|
|
|
|
-h, --help Print this message
|
|
|
|
-D, --log-debug Log debug output
|
|
|
|
-v, --version Print the version
|
|
|
|
-r, --reload Reload cached remote resources
|
|
|
|
--v8-options Print V8 command line options
|
|
|
|
--types Print runtime TypeScript declarations
|
|
|
|
--prefetch Prefetch the dependencies
|
2019-02-11 17:41:13 -05:00
|
|
|
--info Show source file related info
|
2019-02-27 00:14:10 -05:00
|
|
|
--fmt Format code
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
Environment variables:
|
2019-02-27 00:14:10 -05:00
|
|
|
DENO_DIR Set deno's base directory
|
|
|
|
NO_COLOR Set to disable color
|
2019-02-11 17:41:13 -05:00
|
|
|
```
|
|
|
|
|
|
|
|
### Environmental variables
|
|
|
|
|
|
|
|
There are several env vars that control how Deno behaves:
|
|
|
|
|
|
|
|
`DENO_DIR` defaults to `$HOME/.deno` but can be set to any path to control where
|
|
|
|
generated and cached source code is written and read to.
|
|
|
|
|
|
|
|
`NO_COLOR` will turn off color output if set. See https://no-color.org/. User
|
|
|
|
code can test if `NO_COLOR` was set without having `--allow-env` by using the
|
2019-03-03 06:24:33 +09:00
|
|
|
boolean constant `Deno.noColor`.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
### V8 flags
|
|
|
|
|
|
|
|
V8 has many many internal command-line flags, that you can see with
|
|
|
|
`--v8-options`.
|
|
|
|
[It looks like this.](https://gist.github.com/ry/1c5b080dcbdc6367e5612392049c9ee7)
|
|
|
|
|
|
|
|
Particularly useful ones:
|
|
|
|
|
|
|
|
```
|
|
|
|
--async-stack-trace
|
|
|
|
```
|
|
|
|
|
|
|
|
## Internal details
|
|
|
|
|
2019-02-20 16:29:25 +08:00
|
|
|
### Deno and Linux analogy
|
|
|
|
|
|
|
|
| **Linux** | **Deno** |
|
|
|
|
| ------------------------------: | :------------------------------- |
|
|
|
|
| Processes | Web Workers |
|
|
|
|
| Syscalls | Ops |
|
|
|
|
| File descriptors (fd) | [Resource ids (rid)](#resources) |
|
|
|
|
| Scheduler | Tokio |
|
|
|
|
| Userland: libc++ / glib / boost | deno_std |
|
2019-03-03 06:24:33 +09:00
|
|
|
| /proc/\$\$/stat | [Deno.metrics()](#metrics) |
|
2019-02-20 16:29:25 +08:00
|
|
|
| man pages | deno --types |
|
|
|
|
|
|
|
|
#### Resources
|
|
|
|
|
|
|
|
Resources (AKA `rid`) are Deno's version of file descriptors. They are integer
|
|
|
|
values used to refer to open files, sockets, and other concepts. For testing it
|
|
|
|
would be good to be able to query the system for how many open resources there
|
|
|
|
are.
|
|
|
|
|
|
|
|
```ts
|
2019-03-03 06:24:33 +09:00
|
|
|
const { resources, close } = Deno;
|
2019-02-20 16:29:25 +08:00
|
|
|
console.log(resources());
|
|
|
|
// output like: { 0: "stdin", 1: "stdout", 2: "stderr", 3: "repl" }
|
|
|
|
|
|
|
|
// close resource by rid
|
|
|
|
close(3);
|
|
|
|
```
|
|
|
|
|
|
|
|
#### Metrics
|
|
|
|
|
|
|
|
Metrics is deno's internal counters for various statics.
|
|
|
|
|
|
|
|
```ts
|
2019-03-03 06:24:33 +09:00
|
|
|
const { metrics } = Deno;
|
2019-02-20 16:29:25 +08:00
|
|
|
console.log(metrics());
|
|
|
|
// output like: { opsDispatched: 1, opsCompleted: 1, bytesSentControl: 40, bytesSentData: 0, bytesReceived: 176 }
|
|
|
|
```
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
### Schematic diagram
|
|
|
|
|
2019-03-01 16:59:26 +09:00
|
|
|
<img src="images/schematic_v0.2.png">
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
### Profiling
|
|
|
|
|
|
|
|
To start profiling,
|
|
|
|
|
|
|
|
```sh
|
|
|
|
# Make sure we're only building release.
|
|
|
|
export DENO_BUILD_MODE=release
|
|
|
|
# Build deno and V8's d8.
|
|
|
|
./tools/build.py d8 deno
|
|
|
|
# Start the program we want to benchmark with --prof
|
|
|
|
./target/release/deno tests/http_bench.ts --allow-net --prof &
|
|
|
|
# Exercise it.
|
|
|
|
third_party/wrk/linux/wrk http://localhost:4500/
|
|
|
|
kill `pgrep deno`
|
|
|
|
```
|
|
|
|
|
|
|
|
V8 will write a file in the current directory that looks like this:
|
|
|
|
`isolate-0x7fad98242400-v8.log`. To examine this file:
|
|
|
|
|
|
|
|
```sh
|
|
|
|
D8_PATH=target/release/ ./third_party/v8/tools/linux-tick-processor
|
|
|
|
isolate-0x7fad98242400-v8.log > prof.log
|
|
|
|
# on macOS, use ./third_party/v8/tools/mac-tick-processor instead
|
|
|
|
```
|
|
|
|
|
|
|
|
`prof.log` will contain information about tick distribution of different calls.
|
|
|
|
|
|
|
|
To view the log with Web UI, generate JSON file of the log:
|
|
|
|
|
|
|
|
```sh
|
|
|
|
D8_PATH=target/release/ ./third_party/v8/tools/linux-tick-processor
|
|
|
|
isolate-0x7fad98242400-v8.log --preprocess > prof.json
|
|
|
|
```
|
|
|
|
|
|
|
|
Open `third_party/v8/tools/profview/index.html` in your brower, and select
|
|
|
|
`prof.json` to view the distribution graphically.
|
|
|
|
|
|
|
|
To learn more about `d8` and profiling, check out the following links:
|
|
|
|
|
|
|
|
- [https://v8.dev/docs/d8](https://v8.dev/docs/d8)
|
|
|
|
- [https://v8.dev/docs/profile](https://v8.dev/docs/profile)
|
|
|
|
|
|
|
|
### Debugging with LLDB
|
|
|
|
|
|
|
|
We can use LLDB to debug deno.
|
|
|
|
|
|
|
|
```sh
|
|
|
|
lldb -- target/debug/deno tests/worker.js
|
|
|
|
> run
|
|
|
|
> bt
|
|
|
|
> up
|
|
|
|
> up
|
|
|
|
> l
|
|
|
|
```
|
|
|
|
|
|
|
|
To debug Rust code, we can use `rust-lldb`. It should come with `rustc` and is a
|
|
|
|
wrapper around LLDB.
|
|
|
|
|
|
|
|
```sh
|
|
|
|
rust-lldb -- ./target/debug/deno tests/http_bench.ts --allow-net
|
|
|
|
# On macOS, you might get warnings like
|
|
|
|
# `ImportError: cannot import name _remove_dead_weakref`
|
|
|
|
# In that case, use system python by setting PATH, e.g.
|
|
|
|
# PATH=/System/Library/Frameworks/Python.framework/Versions/2.7/bin:$PATH
|
|
|
|
(lldb) command script import "/Users/kevinqian/.rustup/toolchains/1.30.0-x86_64-apple-darwin/lib/rustlib/etc/lldb_rust_formatters.py"
|
|
|
|
(lldb) type summary add --no-value --python-function lldb_rust_formatters.print_val -x ".*" --category Rust
|
|
|
|
(lldb) type category enable Rust
|
|
|
|
(lldb) target create "../deno/target/debug/deno"
|
|
|
|
Current executable set to '../deno/target/debug/deno' (x86_64).
|
|
|
|
(lldb) settings set -- target.run-args "tests/http_bench.ts" "--allow-net"
|
|
|
|
(lldb) b op_start
|
|
|
|
(lldb) r
|
|
|
|
```
|
|
|
|
|
|
|
|
### libdeno
|
|
|
|
|
|
|
|
deno's privileged side will primarily be programmed in Rust. However there will
|
|
|
|
be a small C API that wraps V8 to 1) define the low-level message passing
|
|
|
|
semantics, 2) provide a low-level test target, 3) provide an ANSI C API binding
|
|
|
|
interface for Rust. V8 plus this C API is called "libdeno" and the important
|
|
|
|
bits of the API is specified here:
|
|
|
|
[deno.h](https://github.com/denoland/deno/blob/master/libdeno/deno.h)
|
|
|
|
[libdeno.ts](https://github.com/denoland/deno/blob/master/js/libdeno.ts)
|
|
|
|
|
|
|
|
### Flatbuffers
|
|
|
|
|
|
|
|
We use Flatbuffers to define common structs and enums between TypeScript and
|
|
|
|
Rust. These common data structures are defined in
|
|
|
|
[msg.fbs](https://github.com/denoland/deno/blob/master/src/msg.fbs)
|
|
|
|
|
|
|
|
### Updating prebuilt binaries
|
|
|
|
|
2019-02-16 00:15:44 +08:00
|
|
|
```bash
|
2019-02-11 17:41:13 -05:00
|
|
|
./third_party/depot_tools/upload_to_google_storage.py -b denoland \
|
|
|
|
-e ~/.config/gcloud/legacy_credentials/ry@tinyclouds.org/.boto `which sccache`
|
|
|
|
mv `which sccache`.sha1 prebuilt/linux64/
|
|
|
|
gsutil acl ch -u AllUsers:R gs://denoland/608be47bf01004aa11d4ed06955414e93934516e
|
|
|
|
```
|
|
|
|
|
|
|
|
### Continuous Benchmarks
|
|
|
|
|
|
|
|
https://deno.land/benchmarks.html
|
|
|
|
|
|
|
|
The benchmark chart supposes `//website/data.json` has the type
|
|
|
|
`BenchmarkData[]` where `BenchmarkData` is defined like the below:
|
|
|
|
|
|
|
|
```typescript
|
|
|
|
interface ExecTimeData {
|
|
|
|
mean: number;
|
|
|
|
stddev: number;
|
|
|
|
user: number;
|
|
|
|
system: number;
|
|
|
|
min: number;
|
|
|
|
max: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface BenchmarkData {
|
|
|
|
created_at: string;
|
|
|
|
sha1: string;
|
|
|
|
benchmark: {
|
|
|
|
[key: string]: ExecTimeData;
|
|
|
|
};
|
|
|
|
binarySizeData: {
|
|
|
|
[key: string]: number;
|
|
|
|
};
|
|
|
|
threadCountData: {
|
|
|
|
[key: string]: number;
|
|
|
|
};
|
|
|
|
syscallCountData: {
|
|
|
|
[key: string]: number;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2019-02-27 00:13:49 -05:00
|
|
|
### Logos
|
|
|
|
|
|
|
|
These Deno logos, like the Deno software, are distributed under the MIT license
|
|
|
|
(public domain and free for use)
|
|
|
|
|
2019-03-05 18:57:57 -08:00
|
|
|
- [A hand drawn one by @ry](https://github.com/denoland/deno/blob/master/website/images/deno_logo.png)
|
2019-02-27 00:13:49 -05:00
|
|
|
|
|
|
|
- [An animated one by @hashrock](https://github.com/denolib/animated-deno-logo/)
|
|
|
|
|
|
|
|
- [A high resolution SVG one by @kevinkassimo](https://github.com/denolib/high-res-deno-logo)
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
## Contributing
|
|
|
|
|
|
|
|
[Style Guide](style_guide.html)
|
|
|
|
|
|
|
|
Progress towards future releases is tracked
|
|
|
|
[here](https://github.com/denoland/deno/milestones).
|
|
|
|
|
|
|
|
Please don't make [the benchmarks](https://deno.land/benchmarks.html) worse.
|
|
|
|
|
|
|
|
Ask for help in the [community chat room](https://gitter.im/denolife/Lobby).
|
|
|
|
|
2019-02-14 12:12:11 +11:00
|
|
|
If you are going to work on an issue, mention so in the issue comments _before_
|
|
|
|
you start working on the issue.
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
### Submitting a pull request
|
|
|
|
|
|
|
|
Before submitting, please make sure the following is done:
|
|
|
|
|
2019-02-14 12:12:11 +11:00
|
|
|
1. That there is a related issue and it is referenced in the PR text.
|
|
|
|
2. There are tests that cover the changes.
|
|
|
|
3. Ensure `./tools/test.py` passes.
|
|
|
|
4. Format your code with `tools/format.py`
|
|
|
|
5. Make sure `./tools/lint.py` passes.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
### Changes to `third_party`
|
|
|
|
|
|
|
|
[`deno_third_party`](https://github.com/denoland/deno_third_party) contains most
|
|
|
|
of the external code that Deno depends on, so that we know exactly what we are
|
|
|
|
executing at any given time. It is carefully mantained with a mixture of manual
|
|
|
|
labor and private scripts. It's likely you will need help from @ry or
|
|
|
|
@piscisaureus to make changes.
|
|
|
|
|
|
|
|
### Adding Ops (aka bindings)
|
|
|
|
|
|
|
|
We are very concerned about making mistakes when adding new APIs. When adding an
|
|
|
|
Op to Deno, the counterpart interfaces on other platforms should be researched.
|
|
|
|
Please list how this functionality is done in Go, Node, Rust, and Python.
|
|
|
|
|
2019-03-03 06:24:33 +09:00
|
|
|
As an example, see how `Deno.rename()` was proposed and added in
|
2019-02-11 17:41:13 -05:00
|
|
|
[PR #671](https://github.com/denoland/deno/pull/671).
|
|
|
|
|
|
|
|
### Documenting APIs
|
|
|
|
|
|
|
|
It is important to document public APIs and we want to do that inline with the
|
|
|
|
code. This helps ensure that code and documentation are tightly coupled
|
|
|
|
together.
|
|
|
|
|
|
|
|
#### Utilize JSDoc
|
|
|
|
|
|
|
|
All publicly exposed APIs and types, both via the `deno` module as well as the
|
|
|
|
global/`window` namespace should have JSDoc documentation. This documentation is
|
|
|
|
parsed and available to the TypeScript compiler, and therefore easy to provide
|
|
|
|
further downstream. JSDoc blocks come just prior to the statement they apply to
|
|
|
|
and are denoted by a leading `/**` before terminating with a `*/`. For example:
|
|
|
|
|
|
|
|
```ts
|
|
|
|
/** A simple JSDoc comment */
|
|
|
|
export const FOO = "foo";
|
|
|
|
```
|