Automate code generation by running procedures defined in comment annotations.
This repository is a proof-of-concept for the deno generate
subcommand as
proposed in Deno issue #19176.
To use this tool, add a //deno:generate
comment in your program with the
command you want to run. For example:
//deno:generate deno run https://deno.land/std/examples/cat.ts README.md
To generate code using the CLI tool, run the tool from deno.land/x
with the
command:
deno run -Ar https://deno.land/x/generate/cli/main.ts <entrypoint file>
If you are interested in installing the script, refer to the Installation section.
deno-generate --help
deno-generate <entrypoint file>
You can also define a task in your deno.jsonc
file to run the CLI tool in your
Deno project:
{
"tasks": {
"generate": "deno run -Ar https://deno.land/x/generate/cli/main.ts <entrypoint file>"
}
}
Alternatively, you can install the script as a command and run it locally:
deno install -rf -A https://deno.land/x/generate/cli/main.ts --name=deno-generate
deno-generate <entrypoint file>
Install from script source (Expand for more information)
git clone https://github.com/ethanthatonekid/deno_generate.git
cd deno_generate
deno install -rf -A cli/main.ts --name=deno-generate
The CLI tool only executes //deno:generate
comments that run commands allowed
by the --allow-run
flag. This security measure is in place to prevent the
execution of malicious code. You can set this flag during installation or each
time you run the script.
You can find more information about the --allow-run
flag in the
Deno permissions documentation.
Code generation is an important programming technique because generated files can automate repetitive or complex tasks, improve code consistency and maintainability, and save developers time and effort.
As for use cases, your imagination is the limit. Here are a few:
- Generating code from templates: Developers can define templates for commonly used code patterns and use the CLI tool to automatically generate code that follows those patterns.
- Generating code from schemas: If a project uses a schema to define data models or API specifications, developers can create generators that generate code based on that schema.
- Generating tests: Developers can define test templates that cover common testing scenarios and use the CLI tool to automatically generate tests for their code.
- Generating code from annotations: Developers can add annotations to their code that define which generators to run and how to run them.
The deno generate
subcommand is capable of facilitating the generation of code
in the Deno ecosystem. Developers use code generation to generate code for all
kinds of use cases.
You can find more examples in the examples
directory.
Deno scripts should be able to invoke another Deno in its //deno:generate
statement:
The provided code generates basic TypeScript code using a for loop to export constants.
//deno:generate deno run -A generate.ts
for (let i = 0; i < 10; i++) {
console.log(`export const example${i} = ${i};`);
}
Note While this example works for simple cases, it is advisable to utilize a widely-used library such as ts-morph for more comprehensive and complete examples.
ts-morph
is a TypeScript Compiler API wrapper for static analysis and programmatic code changes. This module provides an extensive set of features and utilities, simplifying the handling of complex code generation scenarios. By leveraging ts-morph, developers can ensure a more robust, maintainable code generation process.
// Create a child process using Deno.Command, running the "generate.ts" script.
const generatorChild = new Deno.Command(Deno.execPath(), {
args: ["run", "generator.ts"],
stdin: "piped",
stdout: "piped",
}).spawn();
// Create another child process, running deno fmt.
const fmtChild = new Deno.Command(Deno.execPath(), {
args: ["fmt", "-"],
stdin: "piped",
stdout: "piped",
}).spawn();
// Pipe the current process stdin to the child process stdin.
generatorChild.stdout.pipeTo(fmtChild.stdin);
// Close the child process stdin.
generatorChild.stdin.close();
// Pipe the child process stdout to a writable file named "generated.ts".
fmtChild.stdout.pipeTo(
Deno.openSync("generated.ts", { write: true, create: true }).writable,
);
OpenAPI is a JSON-based specification that represents comprehensive API details, providing a formal and professional representation of the API specifications. One of the most common use cases of OpenAPI is to generate API clients, simplifying the development process by automatically generating code based on the defined API specifications. OpenAPI schemas offer versatile applications and integrations, enabling a wide range of possibilities for API design and development.
//deno:generate deno run -A npm:[email protected] ./examples/github_api.json -o ./examples/github_api.ts
Lume is a website framework for the Deno ecosystem. Entire static websites are generated by Lume with a single command. See more.
//deno:generate deno run -A https://deno.land/x/[email protected]/cli.ts --src ./examples/lume --dest ./examples/lume/_site
This tool aims to simplify glue code generation for Deno FFI libraries written in Rust. See more.
//deno:generate deno run -A https://deno.land/x/[email protected]/cli.ts
deno-embedder is a tool that simplifies the development and distribution of Deno applications, particularly when access to static files (.txt, .png, etc.) is required at runtime. It allows you to create an embedder.ts file that encompasses both configuration and the main function call, providing benefits such as IDE-based type-checking. See more.
//deno:generate deno run -A embedder.ts
import examplesDir from "../embedder_static/dir.ts";
const exampleFile = await examplesDir.load("with_embedder.ts");
console.log("You are currently reading:", await exampleFile.text());
Bundlee is a deno-embedder alternative.
import { Bundlee } from "https://deno.land/x/bundlee/mod.ts";
//deno:generate deno run -A https://deno.land/x/[email protected]/bundlee.ts --bundle static/ bundle.json
const staticFiles = await Bundlee.load("bundle.json");
//deno:generate deno run -A generate_docs.ts
import doc from "./doc.json" assert { type: "json" };
// Create a child process running `deno doc --json`.
const child = new Deno.Command(Deno.execPath(), {
args: ["doc", "--json"],
stdin: "piped",
stdout: "piped",
}).spawn();
// Pipe the child process stdout to a writable file named "doc.json".
child.stdout.pipeTo(
Deno.openSync("doc.json", { write: true, create: true }).writable,
);
To ensure a consistent developer experience, we recommend following these conventions when using the CLI tool:
To enhance your development workflow, we recommend implementing a pre-commit hook in your project's Git repository. Follow these steps to set it up:
- Create a file named "pre-commit" (without any file extension) within your project's ".git/hooks" directory.
- Ensure that the file is executable by running the following command in your terminal:
chmod +x .git/hooks/pre-commit
- Open the "pre-commit" file in a text editor and add the following code:
#!/bin/bash
# Run generators before committing.
deno task generate
# Check if any files have changed.
git diff --exit-code
When dealing with code generation, there are situations where generated files should not be visible to developers during a pull request. To address this, a setting can be used to differentiate their changes, ensuring a cleaner and more focused code review. On GitHub, you can achieve this by marking specific files with the "linguist-generated" attribute in a ".gitattributes" file1. This attribute allows you to hide these files by default in diffs and exclude them from contributing to the repository language statistics.
To implement this, follow these steps:
- Create a ".gitattributes" file in your project's root directory if it doesn't already exist.
- Open the ".gitattributes" file in a text editor and include the relevant file patterns along with the "linguist-generated" attribute. For example:
# Marking generated files
*.generated.extension linguist-generated
- Save the file and commit it to your repository.
Now, when viewing pull requests or generating diffs on GitHub, the marked files will be hidden by default, providing a more streamlined code review process and excluding them from language statistics calculations.
To run the tool from source, use the following command:
deno run -A cli/main.ts --verbose ./examples/embedder/with_embedder.ts
To run the existing unit tests, use the following command:
deno test
To format your code and check for lint errors, use the following command:
This process cleans your code and identifies common errors.
deno task all
The all
task is defined in deno.jsonc
and executes the
following tasks:
You can run each task individually using the following commands:
deno task fmt
deno task lint
Contributions are welcome! Read the contributing guide for more information.
Using //deno:generate <command>
in a TypeScript file instead of solely relying
on deno task
offers several advantages. Firstly, incorporating
//deno:generate
directly into the source code allows for better integration
between the generators and the codebase. This approach enables developers to
easily understand and manage the generators as an integral part of the project.
Additionally, by utilizing //deno:generate
, we can conveniently run multiple
generators that are closely tied to the specific TypeScript files. While it's
technically feasible to write a deno task
that mimics the functionality of a
series of //deno:generate
commands, this approach may not scale efficiently
when the project relies on numerous generators. By placing the //deno:generate
directives within the relevant TypeScript files, we achieve a more scalable and
flexible solution for managing generator-related tasks.
If I have two files that depend on some generated code, which file do does //deno:generate
belong in?
In situations where multiple files depend on the same generated code, it's highly recommended to create a shared module. This shared module serves as a central location from which both files can import the generated code. By doing so, we promote better code organization and encourage code reuse.
The shared module approach also simplifies future modifications and updates. If the generated code needs to be modified or enhanced, we can make the changes in a single place—the shared module—and have the updates reflect in both files that depend on it. This not only reduces code duplication but also improves maintainability by ensuring consistency throughout the project.
Programmed with 🦕 by @EthanThatOneKid