2019-02-11 17:41:13 -05:00
|
|
|
|
# Deno Style Guide
|
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
## Copyright Headers
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
Most modules in the repository should have the following copyright header:
|
|
|
|
|
|
|
|
|
|
```ts
|
2021-01-12 02:13:41 +09:00
|
|
|
|
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
|
2020-04-04 18:11:43 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
If the code originates elsewhere, ensure that the file has the proper copyright
|
|
|
|
|
headers. We only allow MIT, BSD, and Apache licensed code.
|
|
|
|
|
|
|
|
|
|
## Use underscores, not dashes in filenames.
|
|
|
|
|
|
|
|
|
|
Example: Use `file_server.ts` instead of `file-server.ts`.
|
|
|
|
|
|
|
|
|
|
## Add tests for new features.
|
|
|
|
|
|
|
|
|
|
Each module should contain or be accompanied by tests for its public
|
|
|
|
|
functionality.
|
|
|
|
|
|
|
|
|
|
## TODO Comments
|
|
|
|
|
|
|
|
|
|
TODO comments should usually include an issue or the author's github username in
|
|
|
|
|
parentheses. Example:
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// TODO(ry): Add tests.
|
|
|
|
|
// TODO(#123): Support Windows.
|
|
|
|
|
// FIXME(#349): Sometimes panics.
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## 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.
|
|
|
|
|
|
2020-06-13 19:09:39 +02:00
|
|
|
|
## Inclusive code
|
|
|
|
|
|
|
|
|
|
Please follow the guidelines for inclusive code outlined at
|
|
|
|
|
https://chromium.googlesource.com/chromium/src/+/master/styleguide/inclusive_code.md.
|
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
## Rust
|
|
|
|
|
|
|
|
|
|
Follow Rust conventions and be consistent with existing code.
|
|
|
|
|
|
2020-08-18 01:17:57 +09:00
|
|
|
|
## TypeScript
|
2020-04-04 18:11:43 +01:00
|
|
|
|
|
|
|
|
|
The TypeScript portions of the codebase include `cli/js` for the built-ins and
|
|
|
|
|
the standard library `std`.
|
|
|
|
|
|
|
|
|
|
### Use TypeScript instead of JavaScript.
|
|
|
|
|
|
|
|
|
|
### Use the term "module" instead of "library" or "package".
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### Do not use the filename `index.ts`/`index.js`.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### Exported functions: max 2 args, put the rest into an options object.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
|
|
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:
|
|
|
|
|
|
2020-08-18 01:17:57 +09:00
|
|
|
|
- a distinguishing prototype (e.g. `Array`, `Map`, `Date`, `class MyThing`).
|
2020-07-14 15:24:17 -04:00
|
|
|
|
- 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.
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// BAD: optional parameters not part of options object. (#2)
|
|
|
|
|
export function resolve(
|
|
|
|
|
hostname: string,
|
|
|
|
|
family?: "ipv4" | "ipv6",
|
2020-07-14 15:24:17 -04:00
|
|
|
|
timeout?: number,
|
2019-02-11 17:41:13 -05:00
|
|
|
|
): IPAddress[] {}
|
|
|
|
|
|
|
|
|
|
// GOOD.
|
|
|
|
|
export interface ResolveOptions {
|
|
|
|
|
family?: "ipv4" | "ipv6";
|
|
|
|
|
timeout?: number;
|
|
|
|
|
}
|
|
|
|
|
export function resolve(
|
|
|
|
|
hostname: string,
|
2020-07-14 15:24:17 -04:00
|
|
|
|
options: ResolveOptions = {},
|
2019-02-11 17:41:13 -05:00
|
|
|
|
): 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,
|
2020-07-14 15:24:17 -04:00
|
|
|
|
options: RunShellOptions,
|
2019-02-11 17:41:13 -05:00
|
|
|
|
): string {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// BAD: more than 3 arguments (#1), multiple optional parameters (#2).
|
|
|
|
|
export function renameSync(
|
|
|
|
|
oldname: string,
|
|
|
|
|
newname: string,
|
|
|
|
|
replaceExisting?: boolean,
|
2020-07-14 15:24:17 -04:00
|
|
|
|
followLinks?: boolean,
|
2019-02-11 17:41:13 -05:00
|
|
|
|
) {}
|
|
|
|
|
|
|
|
|
|
// GOOD.
|
|
|
|
|
interface RenameOptions {
|
|
|
|
|
replaceExisting?: boolean;
|
|
|
|
|
followLinks?: boolean;
|
|
|
|
|
}
|
|
|
|
|
export function renameSync(
|
|
|
|
|
oldname: string,
|
|
|
|
|
newname: string,
|
2020-07-14 15:24:17 -04:00
|
|
|
|
options: RenameOptions = {},
|
2019-02-11 17:41:13 -05:00
|
|
|
|
) {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// BAD: too many arguments. (#1)
|
|
|
|
|
export function pwrite(
|
|
|
|
|
fd: number,
|
|
|
|
|
buffer: TypedArray,
|
|
|
|
|
offset: number,
|
|
|
|
|
length: number,
|
2020-07-14 15:24:17 -04:00
|
|
|
|
position: number,
|
2019-02-11 17:41:13 -05:00
|
|
|
|
) {}
|
|
|
|
|
|
|
|
|
|
// BETTER.
|
|
|
|
|
export interface PWrite {
|
|
|
|
|
fd: number;
|
|
|
|
|
buffer: TypedArray;
|
|
|
|
|
offset: number;
|
|
|
|
|
length: number;
|
|
|
|
|
position: number;
|
|
|
|
|
}
|
|
|
|
|
export function pwrite(options: PWrite) {}
|
|
|
|
|
```
|
|
|
|
|
|
2020-10-19 21:41:40 -05:00
|
|
|
|
### Export all interfaces that are used as parameters to an exported member
|
|
|
|
|
|
|
|
|
|
Whenever you are using interfaces that are included in the arguments of an
|
|
|
|
|
exported member, you should export the interface that is used. Here is an
|
|
|
|
|
example:
|
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
// my_file.ts
|
|
|
|
|
export interface Person {
|
|
|
|
|
name: string;
|
|
|
|
|
age: number;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function createPerson(name: string, age: number): Person {
|
|
|
|
|
return { name, age };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// mod.ts
|
|
|
|
|
export { createPerson } from "./my_file.ts";
|
|
|
|
|
export type { Person } from "./my_file.ts";
|
|
|
|
|
```
|
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### Minimize dependencies; do not make circular imports.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
Although `cli/js` and `std` have no external dependencies, we must still be
|
|
|
|
|
careful to keep internal dependencies simple and manageable. In particular, be
|
2020-05-01 03:33:03 -07:00
|
|
|
|
careful not to introduce circular imports.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### If a filename starts with an underscore: `_foo.ts`, do not link to it.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
2020-05-18 06:31:46 -05:00
|
|
|
|
Sometimes there may be situations where an internal module is necessary but its
|
2019-02-11 17:41:13 -05:00
|
|
|
|
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.
|
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### Use JSDoc for exported symbols.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
We strive for complete documentation. Every exported symbol ideally should have
|
|
|
|
|
a documentation line.
|
|
|
|
|
|
2020-08-18 01:17:57 +09:00
|
|
|
|
If possible, use a single line for the JSDoc. Example:
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
```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
|
|
|
|
|
/**
|
2020-04-04 18:11:43 +01:00
|
|
|
|
* Function with non obvious param.
|
|
|
|
|
* @param foo Description of non obvious parameter.
|
2019-03-15 16:53:21 +01:00
|
|
|
|
*/
|
|
|
|
|
```
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
Vertical spacing should be minimized whenever possible. Therefore single line
|
|
|
|
|
comments should be written as:
|
|
|
|
|
|
|
|
|
|
```ts
|
2020-04-04 18:11:43 +01:00
|
|
|
|
/** This is a good single line JSDoc. */
|
2019-02-11 17:41:13 -05:00
|
|
|
|
```
|
|
|
|
|
|
2020-08-18 01:17:57 +09:00
|
|
|
|
And not:
|
2019-02-11 17:41:13 -05:00
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
/**
|
2020-04-04 18:11:43 +01:00
|
|
|
|
* This is a bad single line JSDoc.
|
2019-02-11 17:41:13 -05:00
|
|
|
|
*/
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2020-11-06 17:36:28 +01:00
|
|
|
|
### Resolve linting problems using directives
|
2020-10-06 19:31:36 -05:00
|
|
|
|
|
2020-11-06 17:36:28 +01:00
|
|
|
|
Currently, the building process uses `dlint` to validate linting problems in the
|
|
|
|
|
code. If the task requires code that is non-conformant to linter use
|
2020-12-11 02:45:45 +07:00
|
|
|
|
`deno-lint-ignore <code>` directive to suppress the warning.
|
2020-10-06 19:31:36 -05:00
|
|
|
|
|
|
|
|
|
```typescript
|
2020-11-03 16:19:29 +01:00
|
|
|
|
// deno-lint-ignore no-explicit-any
|
2020-10-06 19:31:36 -05:00
|
|
|
|
let x: any;
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
This ensures the continuous integration process doesn't fail due to linting
|
2020-11-06 17:36:28 +01:00
|
|
|
|
problems, but it should be used scarcely.
|
2020-10-06 19:31:36 -05:00
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### Each module should come with a test module.
|
2019-03-15 23:51:47 +01:00
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
Every module with public functionality `foo.ts` should come with a test module
|
|
|
|
|
`foo_test.ts`. A test for a `cli/js` module should go in `cli/js/tests` due to
|
|
|
|
|
their different contexts, otherwise it should just be a sibling to the tested
|
|
|
|
|
module.
|
2019-03-15 23:51:47 +01:00
|
|
|
|
|
2020-04-04 18:11:43 +01:00
|
|
|
|
### Unit Tests should be explicit.
|
2019-03-15 23:51:47 +01:00
|
|
|
|
|
|
|
|
|
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
|
2020-07-31 11:12:20 +02:00
|
|
|
|
import { assertEquals } from "https://deno.land/std@$STD_VERSION/testing/asserts.ts";
|
2019-03-15 23:51:47 +01:00
|
|
|
|
import { foo } from "./mod.ts";
|
|
|
|
|
|
2020-08-03 17:53:18 +03:00
|
|
|
|
Deno.test("myTestFunction", function () {
|
2019-03-15 23:51:47 +01:00
|
|
|
|
assertEquals(foo(), { bar: "bar" });
|
|
|
|
|
});
|
|
|
|
|
```
|
2020-04-04 18:11:43 +01:00
|
|
|
|
|
|
|
|
|
### Top level functions should not use arrow syntax.
|
|
|
|
|
|
|
|
|
|
Top level functions should use the `function` keyword. Arrow syntax should be
|
|
|
|
|
limited to closures.
|
|
|
|
|
|
2020-08-18 01:17:57 +09:00
|
|
|
|
Bad:
|
2020-04-04 18:11:43 +01:00
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
export const foo = (): string => {
|
|
|
|
|
return "bar";
|
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
2020-08-18 01:17:57 +09:00
|
|
|
|
Good:
|
2020-04-04 18:11:43 +01:00
|
|
|
|
|
|
|
|
|
```ts
|
|
|
|
|
export function foo(): string {
|
|
|
|
|
return "bar";
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### `std`
|
|
|
|
|
|
|
|
|
|
#### Do not depend on external code.
|
|
|
|
|
|
|
|
|
|
`https://deno.land/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.
|
2020-05-25 18:32:34 +01:00
|
|
|
|
|
2020-12-11 02:45:45 +07:00
|
|
|
|
#### Document and maintain browser compatibility.
|
2020-05-25 18:32:34 +01:00
|
|
|
|
|
|
|
|
|
If a module is browser compatible, include the following in the JSDoc at the top
|
|
|
|
|
of the module:
|
|
|
|
|
|
|
|
|
|
```ts
|
2021-01-02 15:17:01 +00:00
|
|
|
|
// This module is browser compatible.
|
2020-05-25 18:32:34 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Maintain browser compatibility for such a module by either not using the global
|
|
|
|
|
`Deno` namespace or feature-testing for it. Make sure any new dependencies are
|
|
|
|
|
also browser compatible.
|