2019-02-11 17:41:13 -05:00
|
|
|
|
# Deno Style Guide
|
|
|
|
|
|
2019-10-24 17:58:57 -04:00
|
|
|
|
## Table of Contents
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
## Use TypeScript
|
|
|
|
|
|
|
|
|
|
## Use the term "module" instead of "library" or "package"
|
|
|
|
|
|
|
|
|
|
For clarity and consistency avoid the terms "library" and "package". Instead use
|
|
|
|
|
"module" to refer to a single JS or TS file and also to refer to a directory of
|
|
|
|
|
TS/JS code.
|
|
|
|
|
|
|
|
|
|
## Do not use the filename `index.ts` nor `index.js`
|
|
|
|
|
|
|
|
|
|
Deno does not treat "index.js" or "index.ts" in a special way. By using these
|
|
|
|
|
filenames, it suggests that they can be left out of the module specifier when
|
|
|
|
|
they cannot. This is confusing.
|
|
|
|
|
|
|
|
|
|
If a directory of code needs a default entry point, use the filename `mod.ts`.
|
|
|
|
|
The filename `mod.ts` follows Rust’s convention, is shorter than `index.ts`, and
|
|
|
|
|
doesn’t come with any preconceived notions about how it might work.
|
|
|
|
|
|
|
|
|
|
## Within `deno_std`, do not depend on external code
|
|
|
|
|
|
|
|
|
|
`deno_std` is intended to be baseline functionality that all Deno programs can
|
|
|
|
|
rely on. We want to guarantee to users that this code does not include
|
|
|
|
|
potentially unreviewed third party code.
|
|
|
|
|
|
|
|
|
|
## Within `deno_std`, minimize dependencies; do not make circular imports.
|
|
|
|
|
|
|
|
|
|
Although `deno_std` is a standalone codebase, we must still be careful to keep
|
|
|
|
|
the internal dependencies simple and manageable. In particular, be careful to
|
|
|
|
|
not to introduce circular imports.
|
|
|
|
|
|
|
|
|
|
## For consistency, use underscores, not dashes in filenames.
|
|
|
|
|
|
|
|
|
|
Example: Instead of `file-server.ts` use `file_server.ts`.
|
|
|
|
|
|
2019-05-16 07:37:47 -07:00
|
|
|
|
## Format code using prettier.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
More specifically, code should be wrapped at 80 columns and use 2-space
|
|
|
|
|
indentation and use camel-case. Use `//format.ts` to invoke prettier.
|
|
|
|
|
|
|
|
|
|
## Exported functions: max 2 args, put the rest into an options object.
|
|
|
|
|
|
|
|
|
|
When designing function interfaces, stick to the following rules.
|
|
|
|
|
|
|
|
|
|
1. A function that is part of the public API takes 0-2 required arguments, plus
|
|
|
|
|
(if necessary) an options object (so max 3 total).
|
|
|
|
|
|
|
|
|
|
2. Optional parameters should generally go into the options object.
|
|
|
|
|
|
|
|
|
|
An optional parameter that's not in an options object might be acceptable if
|
|
|
|
|
there is only one, and it seems inconceivable that we would add more optional
|
|
|
|
|
parameters in the future.
|
|
|
|
|
|
2019-08-28 21:51:36 +08:00
|
|
|
|
<!-- prettier-ignore-start -->
|
|
|
|
|
<!-- see https://github.com/prettier/prettier/issues/3679 -->
|
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
|
3. The 'options' argument is the only argument that is a regular 'Object'.
|
|
|
|
|
|
|
|
|
|
Other arguments can be objects, but they must be distinguishable from a
|
|
|
|
|
'plain' Object runtime, by having either:
|
|
|
|
|
|
2019-08-28 21:51:36 +08:00
|
|
|
|
- a distinguishing prototype (e.g. `Array`, `Map`, `Date`, `class MyThing`)
|
|
|
|
|
- a well-known symbol property (e.g. an iterable with `Symbol.iterator`).
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
This allows the API to evolve in a backwards compatible way, even when the
|
|
|
|
|
position of the options object changes.
|
|
|
|
|
|
2019-08-28 21:51:36 +08:00
|
|
|
|
<!-- prettier-ignore-end -->
|
|
|
|
|
|
2019-02-11 17:41:13 -05:00
|
|
|
|
```ts
|
|
|
|
|
// BAD: optional parameters not part of options object. (#2)
|
|
|
|
|
export function resolve(
|
|
|
|
|
hostname: string,
|
|
|
|
|
family?: "ipv4" | "ipv6",
|
|
|
|
|
timeout?: number
|
|
|
|
|
): IPAddress[] {}
|
|
|
|
|
|
|
|
|
|
// GOOD.
|
|
|
|
|
export interface ResolveOptions {
|
|
|
|
|
family?: "ipv4" | "ipv6";
|
|
|
|
|
timeout?: number;
|
|
|
|
|
}
|
|
|
|
|
export function resolve(
|
|
|
|
|
hostname: string,
|
|
|
|
|
options: ResolveOptions = {}
|
|
|
|
|
): IPAddress[] {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
export interface Environment {
|
|
|
|
|
[key: string]: string;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// BAD: `env` could be a regular Object and is therefore indistinguishable
|
|
|
|
|
// from an options object. (#3)
|
|
|
|
|
export function runShellWithEnv(cmdline: string, env: Environment): string {}
|
|
|
|
|
|
|
|
|
|
// GOOD.
|
|
|
|
|
export interface RunShellOptions {
|
|
|
|
|
env: Environment;
|
|
|
|
|
}
|
|
|
|
|
export function runShellWithEnv(
|
|
|
|
|
cmdline: string,
|
|
|
|
|
options: RunShellOptions
|
|
|
|
|
): string {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// BAD: more than 3 arguments (#1), multiple optional parameters (#2).
|
|
|
|
|
export function renameSync(
|
|
|
|
|
oldname: string,
|
|
|
|
|
newname: string,
|
|
|
|
|
replaceExisting?: boolean,
|
|
|
|
|
followLinks?: boolean
|
|
|
|
|
) {}
|
|
|
|
|
|
|
|
|
|
// GOOD.
|
|
|
|
|
interface RenameOptions {
|
|
|
|
|
replaceExisting?: boolean;
|
|
|
|
|
followLinks?: boolean;
|
|
|
|
|
}
|
|
|
|
|
export function renameSync(
|
|
|
|
|
oldname: string,
|
|
|
|
|
newname: string,
|
|
|
|
|
options: RenameOptions = {}
|
|
|
|
|
) {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// BAD: too many arguments. (#1)
|
|
|
|
|
export function pwrite(
|
|
|
|
|
fd: number,
|
|
|
|
|
buffer: TypedArray,
|
|
|
|
|
offset: number,
|
|
|
|
|
length: number,
|
|
|
|
|
position: number
|
|
|
|
|
) {}
|
|
|
|
|
|
|
|
|
|
// BETTER.
|
|
|
|
|
export interface PWrite {
|
|
|
|
|
fd: number;
|
|
|
|
|
buffer: TypedArray;
|
|
|
|
|
offset: number;
|
|
|
|
|
length: number;
|
|
|
|
|
position: number;
|
|
|
|
|
}
|
|
|
|
|
export function pwrite(options: PWrite) {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## TODO Comments
|
|
|
|
|
|
2019-03-17 17:54:06 +01:00
|
|
|
|
TODO comments should include an issue or the author's github username in
|
2019-02-11 17:41:13 -05:00
|
|
|
|
parentheses. Example:
|
|
|
|
|
|
2019-02-17 13:15:19 +08:00
|
|
|
|
```ts
|
2019-02-11 17:41:13 -05:00
|
|
|
|
// TODO(ry) Add tests.
|
|
|
|
|
// TODO(#123) Support Windows.
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## Copyright headers
|
|
|
|
|
|
|
|
|
|
Most files in `deno_std` should have the following copyright header:
|
|
|
|
|
|
2019-02-17 13:15:19 +08:00
|
|
|
|
```ts
|
2020-01-02 15:13:47 -05:00
|
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
If the code originates elsewhere, ensure that the file has the proper copyright
|
|
|
|
|
headers. We only allow MIT, BSD, and Apache licensed code in `deno_std`.
|
|
|
|
|
|
|
|
|
|
## Top level functions should not use arrow syntax
|
|
|
|
|
|
|
|
|
|
Top level functions should use the `function` keyword. Arrow syntax should be
|
|
|
|
|
limited to closures.
|
|
|
|
|
|
|
|
|
|
Bad
|
|
|
|
|
|
2019-02-17 13:15:19 +08:00
|
|
|
|
```ts
|
2019-06-29 07:49:03 +09:00
|
|
|
|
export const foo = (): string => {
|
2019-02-11 17:41:13 -05:00
|
|
|
|
return "bar";
|
2019-06-29 07:49:03 +09:00
|
|
|
|
};
|
2019-02-11 17:41:13 -05:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Good
|
|
|
|
|
|
2019-02-17 13:15:19 +08:00
|
|
|
|
```ts
|
2019-02-11 17:41:13 -05:00
|
|
|
|
export function foo(): string {
|
|
|
|
|
return "bar";
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## Meta-programming is discouraged. Including the use of Proxy.
|
|
|
|
|
|
|
|
|
|
Be explicit even when it means more code.
|
|
|
|
|
|
|
|
|
|
There are some situations where it may make sense to use such techniques, but in
|
|
|
|
|
the vast majority of cases it does not.
|
|
|
|
|
|
|
|
|
|
## If a filename starts with underscore, do not link to it: `_foo.ts`
|
|
|
|
|
|
|
|
|
|
Sometimes there maybe situations where an internal module is necessary but its
|
|
|
|
|
API is not meant to be stable or linked to. In this case prefix it with an
|
|
|
|
|
underscore. By convention, only files in its own directory should import it.
|
|
|
|
|
|
|
|
|
|
## Use JSDoc to document exported machinery
|
|
|
|
|
|
|
|
|
|
We strive for complete documentation. Every exported symbol ideally should have
|
|
|
|
|
a documentation line.
|
|
|
|
|
|
|
|
|
|
If possible, use a single line for the JS Doc. Example:
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/** foo does bar. */
|
|
|
|
|
export function foo() {
|
|
|
|
|
// ...
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
It is important that documentation is easily human readable, but there is also a
|
|
|
|
|
need to provide additional styling information to ensure generated documentation
|
|
|
|
|
is more rich text. Therefore JSDoc should generally follow markdown markup to
|
|
|
|
|
enrich the text.
|
|
|
|
|
|
|
|
|
|
While markdown supports HTML tags, it is forbidden in JSDoc blocks.
|
|
|
|
|
|
|
|
|
|
Code string literals should be braced with the back-tick (\`) instead of quotes.
|
|
|
|
|
For example:
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/** Import something from the `deno` module. */
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Do not document function arguments unless they are non-obvious of their intent
|
|
|
|
|
(though if they are non-obvious intent, the API should be considered anyways).
|
2019-03-15 16:53:21 +01:00
|
|
|
|
Therefore `@param` should generally not be used. If `@param` is used, it should
|
2019-03-28 11:50:40 +08:00
|
|
|
|
not include the `type` as TypeScript is already strongly typed.
|
2019-03-15 16:53:21 +01:00
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/**
|
|
|
|
|
* Function with non obvious param
|
|
|
|
|
* @param foo Description of non obvious parameter
|
|
|
|
|
*/
|
|
|
|
|
```
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
Vertical spacing should be minimized whenever possible. Therefore single line
|
|
|
|
|
comments should be written as:
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/** This is a good single line JSDoc */
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
And not
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/**
|
|
|
|
|
* This is a bad single line JSDoc
|
|
|
|
|
*/
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Code examples should not utilise the triple-back tick (\`\`\`) notation or tags.
|
|
|
|
|
They should just be marked by indentation, which requires a break before the
|
|
|
|
|
block and 6 additional spaces for each line of the example. This is 4 more than
|
|
|
|
|
the first column of the comment. For example:
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/** A straight forward comment and an example:
|
|
|
|
|
*
|
|
|
|
|
* import { foo } from "deno";
|
|
|
|
|
* foo("bar");
|
|
|
|
|
*/
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Code examples should not contain additional comments. It is already inside a
|
2019-04-21 05:48:48 +03:00
|
|
|
|
comment. If it needs further comments it is not a good example.
|
2019-03-15 23:51:47 +01:00
|
|
|
|
|
|
|
|
|
## Each module should come with tests
|
|
|
|
|
|
|
|
|
|
Each module should come with its test as a sibling with the name
|
|
|
|
|
`modulename_test.ts`. For example the module `foo.ts` should come with its
|
|
|
|
|
sibling `foo_test.ts`.
|
|
|
|
|
|
|
|
|
|
## Unit Tests should be explicit
|
|
|
|
|
|
|
|
|
|
For a better understanding of the tests, function should be correctly named as
|
|
|
|
|
its prompted throughout the test command. Like:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
test myTestFunction ... ok
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Example of test:
|
|
|
|
|
|
|
|
|
|
```ts
|
2019-07-09 18:59:13 -04:00
|
|
|
|
import { assertEquals } from "https://deno.land/std@v0.11/testing/asserts.ts";
|
2019-03-15 23:51:47 +01:00
|
|
|
|
import { foo } from "./mod.ts";
|
|
|
|
|
|
2020-02-11 21:50:20 +01:00
|
|
|
|
Deno.test(function myTestFunction() {
|
2019-03-15 23:51:47 +01:00
|
|
|
|
assertEquals(foo(), { bar: "bar" });
|
|
|
|
|
});
|
|
|
|
|
```
|