From b5d484f6044dd8b28b96914a548af8e0ddfbe517 Mon Sep 17 00:00:00 2001
From: wingbot <109207340+monadabot@users.noreply.github.com>
Date: Tue, 17 Sep 2024 23:57:05 +0200
Subject: [PATCH 1/4] feat(docs): update docs (#1019)
feat(docs): update docs
Updates the Wing docs. See details in [workflow run].
[Workflow Run]: https://github.com/winglang/docsite/actions/runs/10911695728
------
*Automatically created via the "update-docs" workflow*
---
.../04-standard-library/compatibility/compatibility.json | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/api_versioned_docs/version-latest/04-standard-library/compatibility/compatibility.json b/api_versioned_docs/version-latest/04-standard-library/compatibility/compatibility.json
index 044a6234..5846dd52 100644
--- a/api_versioned_docs/version-latest/04-standard-library/compatibility/compatibility.json
+++ b/api_versioned_docs/version-latest/04-standard-library/compatibility/compatibility.json
@@ -366,7 +366,7 @@
},
"signedUrl": {
"sim": {
- "implemented": false,
+ "implemented": true,
"issue": 1383
},
"tf-aws": {
From 08c49a7408055168315570028f6cd1a7ba42d6e0 Mon Sep 17 00:00:00 2001
From: wingbot <109207340+monadabot@users.noreply.github.com>
Date: Wed, 18 Sep 2024 19:36:01 +0200
Subject: [PATCH 2/4] feat(docs): update docs (#1020)
feat(docs): update docs
Updates the Wing docs. See details in [workflow run].
[Workflow Run]: https://github.com/winglang/docsite/actions/runs/10927224919
------
*Automatically created via the "update-docs" workflow*
---
.../version-latest/04-winglibs/04-toc.md | 1 +
.../04-winglibs/05-winglibs/email.md | 126 ++++++++++++++++++
2 files changed, 127 insertions(+)
create mode 100644 versioned_docs/version-latest/04-winglibs/05-winglibs/email.md
diff --git a/versioned_docs/version-latest/04-winglibs/04-toc.md b/versioned_docs/version-latest/04-winglibs/04-toc.md
index e896c479..62c2db3d 100644
--- a/versioned_docs/version-latest/04-winglibs/04-toc.md
+++ b/versioned_docs/version-latest/04-winglibs/04-toc.md
@@ -15,6 +15,7 @@ keywords: [winglib, Wing library]
| [Amazon Cognito](/docs/winglibs/winglibs/cognito) | [@winglibs/cognito](/docs/winglibs/winglibs/cognito) | v0.0.14 | A wing library to work with Amazon Cognito | [sim](/docs/platforms/sim), [tf-aws](/docs/platforms/AWS/tf-aws) |
| [Containers](/docs/winglibs/winglibs/containers) | [@winglibs/containers](/docs/winglibs/winglibs/containers) | v0.1.6 | Deploy containers with Wing | [sim](/docs/platforms/sim), [tf-aws](/docs/platforms/AWS/tf-aws) |
| [Amazon DynamoDB](/docs/winglibs/winglibs/dynamodb) | [@winglibs/dynamodb](/docs/winglibs/winglibs/dynamodb) | v0.2.4 | A Wing library for Amazon DynamoDB | [sim](/docs/platforms/sim), [tf-aws](/docs/platforms/AWS/tf-aws) |
+| [Email](/docs/winglibs/winglibs/email) | [@winglibs/email](/docs/winglibs/winglibs/email) | v0.0.1 | A wing library for sending emails | [sim](/docs/platforms/sim), [tf-aws](/docs/platforms/AWS/tf-aws) |
| [Amazon EventBridge](/docs/winglibs/winglibs/eventbridge) | [@winglibs/eventbridge](/docs/winglibs/winglibs/eventbridge) | v0.1.8 | A Wing library for working with Amazon EventBridge | [sim](/docs/platforms/sim), [tf-aws](/docs/platforms/AWS/tf-aws), [awscdk](/docs/platforms/AWS/awscdk) |
| [FIFO Queue](/docs/winglibs/winglibs/fifoqueue) | [@winglibs/fifoqueue](/docs/winglibs/winglibs/fifoqueue) | v0.0.12 | A wing library to work with FIFO (first-in first-out) Queues | [sim](/docs/platforms/sim), [tf-aws](/docs/platforms/AWS/tf-aws) |
| [GitHub](/docs/winglibs/winglibs/github) | [@winglibs/github](/docs/winglibs/winglibs/github) | v0.0.16 | A wing library to work with GitHub Probot | [*](/docs/platforms/platforms) |
diff --git a/versioned_docs/version-latest/04-winglibs/05-winglibs/email.md b/versioned_docs/version-latest/04-winglibs/05-winglibs/email.md
new file mode 100644
index 00000000..6129564c
--- /dev/null
+++ b/versioned_docs/version-latest/04-winglibs/05-winglibs/email.md
@@ -0,0 +1,126 @@
+---
+title: Email
+id: email
+sidebar_label: Email
+description: A wing library for sending emails
+keywords: [winglib, Wing library]
+---
+# email
+
+## Prerequisites
+
+* [winglang](https://winglang.io).
+
+For AWS, you need to have an AWS account and the AWS CLI installed. You'll also need to have [Terraform](https://developer.hashicorp.com/terraform/install) or [OpenTofu](https://opentofu.org/docs/intro/install/) installed to deploy the application.
+
+## Installation
+
+```sh
+npm i @winglibs/email
+```
+
+## Usage
+
+```js
+bring email;
+
+let email = new email.Email(sender: "example@example.com");
+
+new cloud.Function(inflight () => {
+ email.send(
+ to: ["example@example.com"],
+ subject: "My subject",
+ text: "My content",
+ html: "
My content
", // optional
+ );
+});
+```
+
+### Simulator
+
+When using `email.Email` in the local simulator, emails are mocked and are emitted to the logs.
+A table showing all emails that have been sent can be viewed in the email resource's interaction panel.
+
+### AWS
+
+When compiled to AWS platforms, the email resource uses [Amazon SES](https://aws.amazon.com/ses/).
+For testing, we recommend using your own email address for `sender` since sender email addresses must be verified.
+When the application is deployed, an email will be sent to verify the configured `sender` address.
+
+By default, new AWS accounts are in the sandbox mode. This means emails can only be sent to verified addresses. It also limits the number of emails that can be sent. To send emails to other addresses, you need to request production access [here](https://docs.aws.amazon.com/ses/latest/dg/request-production-access.html).
+
+## License
+
+This library is licensed under the [MIT License](./LICENSE).
+
+---
+## API Reference
+
+### Table of Contents
+
+- **Classes**
+ - Email
+- **Interfaces**
+ - IEmail
+- **Structs**
+ - EmailProps
+ - SendEmailOptions
+
+### Email (preflight class)
+
+*No description*
+
+#### Constructor
+
+```
+new(props: EmailProps): Email
+```
+
+#### Properties
+
+*No properties*
+
+#### Methods
+
+| **Signature** | **Description** |
+| --- | --- |
+| inflight send(options: SendEmailOptions): void
| *No description* |
+
+### IEmail (interface)
+
+*No description*
+
+#### Properties
+
+*No properties*
+
+#### Methods
+
+| **Signature** | **Description** |
+| --- | --- |
+| inflight send(options: SendEmailOptions): void
| Sends a simple email. |
+
+### EmailProps (struct)
+
+*No description*
+
+#### Properties
+
+| **Name** | **Type** | **Description** |
+| --- | --- | --- |
+| sender
| str
| The email address for the sender of all emails. |
+
+### SendEmailOptions (struct)
+
+*No description*
+
+#### Properties
+
+| **Name** | **Type** | **Description** |
+| --- | --- | --- |
+| html
| str?
| The body of the email, in HTML. @default - The text body will be used as the HTML body. |
+| subject
| str
| The subject of the email. |
+| text
| str
| The body of the email, in plain text. |
+| to
| Array
| The email addresses to send the email to. |
+
+
From cfc5f55fdfce5481a25dd1be48ae327074ba7454 Mon Sep 17 00:00:00 2001
From: wingbot <109207340+monadabot@users.noreply.github.com>
Date: Thu, 19 Sep 2024 18:32:48 -0400
Subject: [PATCH 3/4] feat(docs): update docs (#1021)
feat(docs): update docs
Updates the Wing docs. See details in [workflow run].
[Workflow Run]: https://github.com/winglang/docsite/actions/runs/10949868741
------
*Automatically created via the "update-docs" workflow*
---
.../version-latest/05-language-reference.md | 149 ++++++++++++++++++
1 file changed, 149 insertions(+)
diff --git a/api_versioned_docs/version-latest/05-language-reference.md b/api_versioned_docs/version-latest/05-language-reference.md
index 6433997d..f3240ad4 100644
--- a/api_versioned_docs/version-latest/05-language-reference.md
+++ b/api_versioned_docs/version-latest/05-language-reference.md
@@ -1280,10 +1280,159 @@ let rawData: bytes = fs.readBytes("path/to/file");
fs.writeBytes("path/to/file", rawData);
```
+#### Roadmap
+
+The following features are not yet implemented, but we are planning to add them in the future:
+
+* `mutbytes` - see https://github.com/winglang/wing/issues/7144
+
[`▲ top`][top]
---
+### 1.15 Type reflection
+
+Wing supports reflection on types in preflight code. This allows you to inspect and analyze the structure of types at compile-time, which can be useful for generating code, validating type constraints, or building generic utilities.
+
+Using reflection, you can obtain information about:
+- The methods of a type
+- The fields of a struct or class
+- The variants of an enum
+- The implemented interfaces of a class
+
+#### 1.15.1 Basic usage
+
+The `@type` intrinsic function is used to obtain type information:
+
+```TS
+let t: Type = @type(MyStruct);
+let t: Type = @type(IMyInterface);
+```
+
+The `Type` value has a `kind` field which indicates the general category of the type.
+
+```TS
+let t = @type(MyStruct);
+log(t.kind); // "struct", "class", "interface", "num", "str", "bool", etc.
+```
+
+#### 1.15.2 Specific type information
+
+The `Type` value can be converted to more specific type representations:
+
+```TS
+let t = @type(SomeType);
+
+if let st = t.asStruct() {
+ log(st.name);
+ for field in st.fields {
+ log(field.name);
+ log(field.type.kind);
+ }
+}
+
+if let cl = t.asClass() {
+ log(cl.name);
+ for method in cl.methods {
+ log(method.name);
+ log(method.returnType.kind);
+ }
+}
+
+if let en = t.asEnum() {
+ log(en.name);
+ for variant in en.variants {
+ log(variant);
+ }
+}
+```
+
+#### 1.15.3 Advanced examples
+
+Here are some more advanced examples of using preflight reflection:
+
+```TS
+// Generate a JSON schema for a struct
+let generateJsonSchema = (structType: Type): str => {
+ if let st = structType.asStruct() {
+ let schema = MutJson {
+ type: "object",
+ properties: {},
+ required: []
+ };
+
+ for field in st.fields {
+ let fieldSchema = {};
+ if field.type.kind == "str" {
+ fieldSchema.type = "string";
+ } else if field.type.kind == "num" {
+ fieldSchema.type = "number";
+ } // ... handle other types
+
+ schema.properties[field.name] = fieldSchema;
+ if !field.optional {
+ schema.required.push(field.name);
+ }
+ }
+
+ return Json.stringify(schema);
+ }
+ throw "Input must be a struct type";
+};
+
+struct User {
+ name: str;
+ age: num;
+ email: str?;
+}
+
+log(generateJsonSchema(@type(User)));
+
+// Check if a class implements an interface
+let implementsInterface = (classType: Type, interfaceType: Type): bool => {
+ if let cl = classType.asClass() {
+ if let int = interfaceType.asInterface() {
+ for impl in cl.implements {
+ if impl.name == int.name {
+ return true;
+ }
+ }
+ }
+ }
+ return false;
+};
+
+interface ILogger {
+ log(message: str): void;
+}
+
+class ConsoleLogger impl ILogger {
+ log(message: str) {
+ // ...
+ }
+}
+
+assert(implementsInterface(@type(ConsoleLogger), @type(ILogger)));
+```
+
+#### 1.15.4 Implementation sketch
+
+The preflight reflection system could be implemented as follows:
+
+1. During compilation, the Wing compiler builds a type information database for all types encountered in the code.
+
+2. The `@type` function is implemented as a compiler intrinsic that looks up the type information and emits a `Type` object into the generated JavaScript code.
+
+3. The `Type` class and its variants (`StructType`, `ClassType`, etc.) are implemented as special compiler-known types that provide an API for accessing the type information.
+
+5. The methods on `Type` (like `asStruct()`, `asClass()`, etc.) are implemented to return the appropriate specific type object if the conversion is valid, or `nil` if not.
+
+6. All of this happens at compile-time, so there's no runtime overhead for using reflection. The compiler can optimize away any reflection code that isn't used to affect the runtime behavior of the program.
+
+This implementation allows for powerful compile-time introspection while maintaining Wing's performance characteristics and type safety.
+
+[`▲ top`][top]
+
## 2. Statements
### 2.1 bring
From 0c7df78d0b4eff071b0dcb0d628de71000800e9e Mon Sep 17 00:00:00 2001
From: David Boyne
Date: Mon, 23 Sep 2024 10:11:57 +0100
Subject: [PATCH 4/4] fix(docs): fixed platform core concept page (#1022)
Reviewed core concept for platforms, but we already have this
information in another page, so just refactoring some pages.
Move intro to platforms to core concepts page and new redirect for old
page.
Co-authored-by: David Boyne
---
redirects.ts | 2 +
.../02-concepts/03-platforms.md | 338 ++++++++++++++++-
.../01-understanding-platforms.md | 345 ------------------
3 files changed, 339 insertions(+), 346 deletions(-)
delete mode 100644 versioned_docs/version-latest/03-platforms/01-understanding-platforms.md
diff --git a/redirects.ts b/redirects.ts
index 221cca27..a592738d 100644
--- a/redirects.ts
+++ b/redirects.ts
@@ -102,6 +102,8 @@ export default [
{ to: '/docs/api/category/util', from: ['/docs/category/util'] },
{ to: '/docs/api/standard-library', from: ['/docs/category/standard-library'] },
+ { to: '/docs/concepts/platforms', from: ['/docs/platforms/platforms'] },
+
// redirects for wing.learn section
{ to: "https://learn.winglang.io/learn", from: ["/learn/"] },
{ to: "https://learn.winglang.io/learn/preflight-inflight", from: ["/learn/preflight-inflight"] },
diff --git a/versioned_docs/version-latest/02-concepts/03-platforms.md b/versioned_docs/version-latest/02-concepts/03-platforms.md
index 98d7ef3d..652337e5 100644
--- a/versioned_docs/version-latest/02-concepts/03-platforms.md
+++ b/versioned_docs/version-latest/02-concepts/03-platforms.md
@@ -5,5 +5,341 @@ description: Wing platforms
keywords: [platforms, targets, target, platform, aws, gcp, azure, sim, terraform, cloudformation, pulumi, provisioning engines, multi-cloud]
---
-TODO: High level summary of platforms? Linking to the platforms section we have?
+When working with the Wing programming language, an integral part of the compilation process is the use of platform. In essence, platform specify how and where your application is deployed. They determine both the cloud environment and the provisioning engine that the code will be deployed with.
+You can view the list of available builtin platform with the `wing compile --help` command. Here is an example of the output:
+
+```sh
+wing compile --help
+Usage: wing compile [options] [entrypoint]
+
+Compiles a Wing program
+
+Arguments:
+ entrypoint program .w entrypoint
+
+Options:
+ -t, --platform --platform Target platform provider (builtin: sim, tf-aws, tf-azure, tf-gcp) (default: [sim])
+ -h, --help display help for command
+```
+
+Wing is shipped with several builtin platforms such as `sim`, `tf-aws`, `tf-azure`, and `tf-gcp` but it is also possible to create and use [custom platforms](#custom-platforms) to fully control how Wing resources are deployed to the cloud.
+
+These providers contain a combination of provision engine and cloud environment in their names, we refer to these as the platform target (which is discussed in more detail below). The only exception is `sim`, which is a special platform for testing and simulating applications locally.
+
+### Specifying Multiple Platforms
+
+You may have noticed that the `--platform` option can be provided multiple times. This means you can specify multiple platforms to compile your application to. For example, if you wanted to compile your application using multiple platforms
+
+```sh
+wing compile app.main.w --platform tf-aws --platform platform-foo --platform platform-bar
+```
+The order in which platforms are evaluated is important.
+
+The first platform in the list is the primary platform, it is responsible for providing the Wing compiler with the base App that will be used to determine how resources are created, as well it will also lay the ground work for what target the rest of the platforms will need to be compatible with.
+
+#### Implicit Platforms
+
+Additionally, you can use naming conventions to implicitly define platforms that should be used. These platform files can be located in the root of your project or in a library that your project uses. The naming convention is as follows:
+```sh
+wplatform.js
+*.wplatform.js
+```
+
+For example, if you have a file named `custom.wplatform.js` in the root of your project, it will automatically be added to the list of platforms to be used when compiling your application. Its also important to note that implicit platforms are always loaded after the platforms specified in the `--platform` option.
+
+The use of implicit platforms can be beneficial when writing a Wing library that requires a specific platform to be used. For example, if you are writing a library that requires a specific parameter to be passed to the platform, you can use an implicit platform to ensure that the parameter is always provided.
+
+For example, if your library structure looks like this:
+
+```sh
+my-library/
+ lib.w
+ custom.wplatform.js
+```
+
+Then the custom platform can define any required parameters that the library needs to function properly. (see [Defining Custom Platform Parameters](#defining-custom-platform-parameters) for more information on how to define custom platform parameters)
+
+
+### Provisioning Engines
+
+Provisioning is the process of setting up and creating infrastructure, and the provisioning engine is the driver behind this deployment. Common engines used in the Wing compilation process include Terraform and AWS CDK, with support for more planned ([tracking issue](https://github.com/winglang/wing/issues/2066)).
+
+Understanding the differences between provisioning engines will help as we dive deeper into the additional concepts of the platform provider system.
+
+### Platform Targets
+
+Platform targets are the combination of a provisioning engine and a cloud environment. For example, `tf-aws` is a platform target that uses Terraform as the provisioning engine and AWS as the cloud environment.
+
+It is worth noting that the platform names are not guaranteed to match their targets, we will see this more as we delve into the idea of Custom Platforms below.
+
+Though not currently implemented, the platform target system is designed with extensibility in mind, as it will be used to determine compatibility between different platforms ([tracking issue](https://github.com/winglang/wing/issues/1474))
+
+#### Platform Parameters
+
+Some platform targets may require additional parameters to be provided. These parameters can be used to pass configuration values to the platform. For example, the platform target `tf-aws` has an optional parameter that can be specified to determine if a new VPC
+should be created or if an existing VPC should be used. In order to provide this parameter, you can use the `--value` option in the cli to specify a key-value pair. For example:
+
+```sh
+wing compile app.main.w --platform tf-aws --value tf-aws/vpc=existing
+```
+
+which will tell the `tf-aws` platform to use an existing VPC. However this will result in an parameter validation error, since when using an existing VPC, you will be required to add additional parameters such as `vpcId` and subnets. As shown in the error below:
+
+```sh
+Error: Parameter validation errors:
+- must have required property 'vpc_id'
+- must have required property 'private_subnet_ids'
+- must have required property 'public_subnet_ids'
+```
+
+it is possible to provide these additional parameters using the `--value` option as well. For example:
+
+```sh
+wing compile app.main.w --platform tf-aws --value tf-aws/vpc=existing --value tf-aws/vpcId=vpc-1234567890 --value tf-aws/privateSubnetId=subnet-1234567890 --value tf-aws/publicSubnetId=subnet-1234567890
+```
+
+Though this may be a bit verbose. As an alternative you can use a values file. Value files are a way to provide multiple config values in a single file. By default the compiler will look for a file named `wing` with any of the following extensions `.json`, `.yaml`, `.yml`, `.toml` Though you can also specify a custom file using the `--values` option.
+
+Here is an example of using a `wing.toml` file to provide the same parameters as above:
+
+```toml
+[ tf-aws ]
+# vpc can be set to "new" or "existing"
+vpc = "new"
+# vpc_lambda will ensure that lambda functions are created within the vpc on the private subnet
+vpc_lambda = true
+# vpc_api_gateway will ensure that the api gateway is created within the vpc on the private subnet
+vpc_api_gateway = true
+# The following parameters will be required if using "existing" vpc
+# vpc_id = "vpc-123xyz"
+# private_subnet_ids = ["subnet-123xyz"]
+# public_subnet_ids = ["subnet-123xyz"]
+```
+
+#### Target-specific code
+
+There might be times when you need to write code that is specific to a particular platform target. For example, you may want to activate a verbose logging service only when testing locally to save on cloud log storage costs.
+
+With the Wing `util` library, you can access environment variables. The `WING_TARGET` environment variable contains the current platform target as it's value, which you can use to conditionally run target-specific code. See the example below:
+
+```js playground example
+bring cloud;
+bring util;
+
+let invocationCounter = new cloud.Counter();
+let queue = new cloud.Queue();
+
+queue.setConsumer(inflight (msg: str) => {
+ invocationCounter.inc();
+});
+
+new cloud.Function(inflight ()=> {
+ // push a message to queue
+ queue.push("m");
+ // sleep according to target
+ if util.env("WING_TARGET") == "sim" {
+ log("Running on Simulator, sleeping for 1s");
+ util.sleep(1s);
+ } else {
+ log("Running on the cloud, sleeping for 30s");
+ util.sleep(30s);
+ }
+ log("Function invoked {invocationCounter.peek()} times");
+});
+
+```
+
+## Custom Platforms
+
+Wing's platform architecture is not just limited to the built-in platforms that come with the language; it's extensible. This means you can create custom platforms tailored to your needs, whether to support a unique cloud provider, introduce additional optimization layers, or integrate with specific enterprise systems.
+
+### Why might you want to create a custom platform?
+
+There are many reasons why you might want to create a custom platform. Here are a few examples:
+
+- Custom infrastructure requirements: Wing's built-in platforms are opinionated about how various resources are implemented. With custom platforms, you can fully control the infrastructure configuration of each cloud resource based on the needs, policies and constraints of your team.
+
+- Enhanced Security: Some organizations have stringent security requirements. With custom platforms, you can embed additional security checks, audits, or encryption layers to suit these needs.
+
+- Optimizations: Your organization may have developed optimization strategies that can help reduce costs or improve performance when deploying applications. Integrating these strategies into a custom platform can make them a seamless part of the deployment process.
+
+### Creating a custom platform
+
+Developing a custom platform requires understanding and adhering to the `IPlatform` interface. This ensures that your custom platform can be integrated smoothly with the Wing compilation process.
+
+The IPlatform interface is defined as follows:
+
+```ts
+export interface IPlatform {
+ // Define the target compatibility of the platform
+ readonly target: string;
+
+ // Define the App that will be used for creating resources
+ newApp?(appProps: AppProps): App;
+
+ // Define overrides for concrete resources
+ newInstance?(type: string, scope: Construct, id: string, props: any): any;
+
+ // Synthesis Hooks
+ preSynth?(app: Construct): void;
+ postSynth?(config: any): any;
+ validate?(config: any): any;
+}
+```
+
+### Using a custom platform
+
+When running the `wing compile` command, the `--platform` option is used to specify the platform provider(s) you wish to use. This option accepts variadic arguments, which means you can specify any number of platforms.
+
+The specified platform can be a built-in platform, or a path to a custom platform. For example, if you have a custom platform named `my-platform`, you can specify it as follows:
+
+```sh
+wing compile --platform tf-aws --platform ../my-platform
+```
+
+### Synthesis Hooks
+
+In the above interface there are three methods that are categorized as synthesis hooks. These hooks are called by the compiler at various points during the compilation process. They allow you to hook into the compilation process and apply customizations.
+
+Your platform only needs to implement the methods that are relevant to your use case. For example, if you are creating a platform that is designed to apply additional security configurations for your organization, then you may only need to implement the `preSynth` hook.
+
+Lets take a look at what each hook is responsible for:
+
+:::info Examples Incoming
+The following examples of hooks use simple JavaScript files for brevity. However, you can and probably would want to build your platform as a Node library to package and distribute it. Examples of this are coming soon.
+:::
+
+### `preSynth` hook
+
+API Reference
+```ts
+preSynth(app: Construct): void;
+```
+
+This hook is called before the compiler begins to synthesize. In the context of a
+Terraform-based platform like `tf-aws`, this hook will have access to the root
+node of the construct tree. This allows the platform to add or change [CDK for
+Terraform](https://github.com/hashicorp/terraform-cdk) constructs before the
+tree is synthesized.
+
+The following example adds a bucket to the root node.
+```js
+const s3_bucket = require("@cdktf/provider-aws/lib/s3-bucket");
+
+exports.Platform = class MyPlatform {
+ preSynth(app) {
+ // app is the root node of the construct tree
+ new s3_bucket.S3Bucket(app, "MyPlatformBucket", {
+ bucket: "my-platform-bucket",
+ });
+ }
+}
+```
+
+### `postSynth` hook
+
+API Reference
+```ts
+postSynth(config: any): any;
+```
+
+This hook runs after artifacts were synthesized. When compiling to a
+Terraform-based platform like `tf-aws`, the hook will have access
+to the raw Terraform JSON configuration, allowing for manipulation of the JSON
+that is written to the compiled output directory.
+
+This hook is useful for adding customizations that can not be applied in the
+context of the preSynth hook. Its worth noting that this is not meant as a
+validation phase since the config is still mutable by subsequent platforms.
+
+The following example manipulates the Terraform configuration to use a S3
+backend. For brevity, the example uses hard coded values.
+```js
+exports.Platform = class MyPlatform {
+ postSynth(config) {
+ config.terraform.backend = {
+ s3: {
+ bucket: "my-wing-state-bucket",
+ key: "platforms-rock.tfstate",
+ region: "us-east-1",
+ }
+ }
+ return config;
+ }
+}
+```
+
+### `validate` hook
+
+API Reference
+```ts
+validate(config: any): void;
+```
+
+This hook is called right after the `postSynth` hook and provided the same
+context object. In the context of a Terraform-based platform like `tf-aws`, this
+is the same Terraform JSON configuration. However, does not allow configuration
+to be mutated, which allows platforms to validate the configuration without
+worrying about another platform mutating after the fact.
+
+The following example validates that buckets all have versioning enabled
+and throw an error during compilation if they don't.
+```js
+exports.Platform = class MyPlatform {
+ validate(config) {
+ for (const bucketEntry of Object.keys(config.resource.aws_s3_bucket)) {
+ const bucket = config.resource.aws_s3_bucket[bucketEntry];
+ if (!bucket.versioning.enabled) {
+ throw new Error(`Bucket ${bucketEntry} does not have versioning enabled`);
+ }
+ }
+ }
+}
+```
+
+### Defining Custom Platform Parameters
+
+In addition to the hooks mentioned above, you can also define custom parameters for your platform. These parameters can be used to pass
+configuration to your platform and can be optional or required.
+
+Parameters are defined using the `parameters` property of the platform. These parameters are expected to be provided in the form of a
+(JSON schema)[https://json-schema.org/].
+
+The following example shoes how to define three parameters `replicateAllBuckets`, `bucketsToReplicate` and `replicaRegion` for a custom platform.
+Our platform's logic will either replicate all buckets if `replicateAllBuckets` is set to `true` or replicate only the buckets specified in `bucketsToReplicate` to the region specified in `replicaRegion`.
+
+Its important to understand the relationship between these parameters, as in if `replicateAllBuckets` is set to `true` then `bucketsToReplicate` is not required.
+Whereas no matter the value of `replicateAllBuckets`, `replicaRegion` is always required.
+
+Luckily, JSON schema allows us to define these relationships and constraints like so:
+
+```js
+class MyPlatform {
+ parameters = {
+ type: "object",
+ required: ["replicateAllBuckets", "replicaRegion"],
+ properties: {
+ replicateAllBuckets: {
+ type: "boolean",
+ },
+ nameOfBucketsToReplicate: {
+ type: "array",
+ items: {
+ type: "string",
+ },
+ },
+ replicaRegion: {
+ type: "string",
+ },
+ },
+ "$comment": "Here in an allOf we can define multiple conditions that must be met for the schema to be valid",
+ allOf: [
+ {
+ if: { properties: { replicateAllBuckets: { const: false }} },
+ then: { required: ["nameOfBucketsToReplicate"] },
+ }
+ ]
+ }
+}
+```
\ No newline at end of file
diff --git a/versioned_docs/version-latest/03-platforms/01-understanding-platforms.md b/versioned_docs/version-latest/03-platforms/01-understanding-platforms.md
deleted file mode 100644
index 52b5c7db..00000000
--- a/versioned_docs/version-latest/03-platforms/01-understanding-platforms.md
+++ /dev/null
@@ -1,345 +0,0 @@
----
-title: What are platforms?
-id: platforms
-description: Wing platforms
-keywords: [platforms, targets, target, platform, aws, gcp, azure, sim, terraform, cloudformation, pulumi, provisioning engines, multi-cloud]
----
-
-When working with the Wing programming language, an integral part of the compilation process is the use of platform. In essence, platform specify how and where your application is deployed. They determine both the cloud environment and the provisioning engine that the code will be deployed with.
-
-You can view the list of available builtin platform with the `wing compile --help` command. Here is an example of the output:
-
-```sh
-wing compile --help
-Usage: wing compile [options] [entrypoint]
-
-Compiles a Wing program
-
-Arguments:
- entrypoint program .w entrypoint
-
-Options:
- -t, --platform --platform Target platform provider (builtin: sim, tf-aws, tf-azure, tf-gcp) (default: [sim])
- -h, --help display help for command
-```
-
-Wing is shipped with several builtin platforms such as `sim`, `tf-aws`, `tf-azure`, and `tf-gcp` but it is also possible to create and use [custom platforms](#custom-platforms) to fully control how Wing resources are deployed to the cloud.
-
-These providers contain a combination of provision engine and cloud environment in their names, we refer to these as the platform target (which is discussed in more detail below). The only exception is `sim`, which is a special platform for testing and simulating applications locally.
-
-### Specifying Multiple Platforms
-
-You may have noticed that the `--platform` option can be provided multiple times. This means you can specify multiple platforms to compile your application to. For example, if you wanted to compile your application using multiple platforms
-
-```sh
-wing compile app.main.w --platform tf-aws --platform platform-foo --platform platform-bar
-```
-The order in which platforms are evaluated is important.
-
-The first platform in the list is the primary platform, it is responsible for providing the Wing compiler with the base App that will be used to determine how resources are created, as well it will also lay the ground work for what target the rest of the platforms will need to be compatible with.
-
-#### Implicit Platforms
-
-Additionally, you can use naming conventions to implicitly define platforms that should be used. These platform files can be located in the root of your project or in a library that your project uses. The naming convention is as follows:
-```sh
-wplatform.js
-*.wplatform.js
-```
-
-For example, if you have a file named `custom.wplatform.js` in the root of your project, it will automatically be added to the list of platforms to be used when compiling your application. Its also important to note that implicit platforms are always loaded after the platforms specified in the `--platform` option.
-
-The use of implicit platforms can be beneficial when writing a Wing library that requires a specific platform to be used. For example, if you are writing a library that requires a specific parameter to be passed to the platform, you can use an implicit platform to ensure that the parameter is always provided.
-
-For example, if your library structure looks like this:
-
-```sh
-my-library/
- lib.w
- custom.wplatform.js
-```
-
-Then the custom platform can define any required parameters that the library needs to function properly. (see [Defining Custom Platform Parameters](#defining-custom-platform-parameters) for more information on how to define custom platform parameters)
-
-
-### Provisioning Engines
-
-Provisioning is the process of setting up and creating infrastructure, and the provisioning engine is the driver behind this deployment. Common engines used in the Wing compilation process include Terraform and AWS CDK, with support for more planned ([tracking issue](https://github.com/winglang/wing/issues/2066)).
-
-Understanding the differences between provisioning engines will help as we dive deeper into the additional concepts of the platform provider system.
-
-### Platform Targets
-
-Platform targets are the combination of a provisioning engine and a cloud environment. For example, `tf-aws` is a platform target that uses Terraform as the provisioning engine and AWS as the cloud environment.
-
-It is worth noting that the platform names are not guaranteed to match their targets, we will see this more as we delve into the idea of Custom Platforms below.
-
-Though not currently implemented, the platform target system is designed with extensibility in mind, as it will be used to determine compatibility between different platforms ([tracking issue](https://github.com/winglang/wing/issues/1474))
-
-#### Platform Parameters
-
-Some platform targets may require additional parameters to be provided. These parameters can be used to pass configuration values to the platform. For example, the platform target `tf-aws` has an optional parameter that can be specified to determine if a new VPC
-should be created or if an existing VPC should be used. In order to provide this parameter, you can use the `--value` option in the cli to specify a key-value pair. For example:
-
-```sh
-wing compile app.main.w --platform tf-aws --value tf-aws/vpc=existing
-```
-
-which will tell the `tf-aws` platform to use an existing VPC. However this will result in an parameter validation error, since when using an existing VPC, you will be required to add additional parameters such as `vpcId` and subnets. As shown in the error below:
-
-```sh
-Error: Parameter validation errors:
-- must have required property 'vpc_id'
-- must have required property 'private_subnet_ids'
-- must have required property 'public_subnet_ids'
-```
-
-it is possible to provide these additional parameters using the `--value` option as well. For example:
-
-```sh
-wing compile app.main.w --platform tf-aws --value tf-aws/vpc=existing --value tf-aws/vpcId=vpc-1234567890 --value tf-aws/privateSubnetId=subnet-1234567890 --value tf-aws/publicSubnetId=subnet-1234567890
-```
-
-Though this may be a bit verbose. As an alternative you can use a values file. Value files are a way to provide multiple config values in a single file. By default the compiler will look for a file named `wing` with any of the following extensions `.json`, `.yaml`, `.yml`, `.toml` Though you can also specify a custom file using the `--values` option.
-
-Here is an example of using a `wing.toml` file to provide the same parameters as above:
-
-```toml
-[ tf-aws ]
-# vpc can be set to "new" or "existing"
-vpc = "new"
-# vpc_lambda will ensure that lambda functions are created within the vpc on the private subnet
-vpc_lambda = true
-# vpc_api_gateway will ensure that the api gateway is created within the vpc on the private subnet
-vpc_api_gateway = true
-# The following parameters will be required if using "existing" vpc
-# vpc_id = "vpc-123xyz"
-# private_subnet_ids = ["subnet-123xyz"]
-# public_subnet_ids = ["subnet-123xyz"]
-```
-
-#### Target-specific code
-
-There might be times when you need to write code that is specific to a particular platform target. For example, you may want to activate a verbose logging service only when testing locally to save on cloud log storage costs.
-
-With the Wing `util` library, you can access environment variables. The `WING_TARGET` environment variable contains the current platform target as it's value, which you can use to conditionally run target-specific code. See the example below:
-
-```js playground example
-bring cloud;
-bring util;
-
-let invocationCounter = new cloud.Counter();
-let queue = new cloud.Queue();
-
-queue.setConsumer(inflight (msg: str) => {
- invocationCounter.inc();
-});
-
-new cloud.Function(inflight ()=> {
- // push a message to queue
- queue.push("m");
- // sleep according to target
- if util.env("WING_TARGET") == "sim" {
- log("Running on Simulator, sleeping for 1s");
- util.sleep(1s);
- } else {
- log("Running on the cloud, sleeping for 30s");
- util.sleep(30s);
- }
- log("Function invoked {invocationCounter.peek()} times");
-});
-
-```
-
-## Custom Platforms
-
-Wing's platform architecture is not just limited to the built-in platforms that come with the language; it's extensible. This means you can create custom platforms tailored to your needs, whether to support a unique cloud provider, introduce additional optimization layers, or integrate with specific enterprise systems.
-
-### Why might you want to create a custom platform?
-
-There are many reasons why you might want to create a custom platform. Here are a few examples:
-
-- Custom infrastructure requirements: Wing's built-in platforms are opinionated about how various resources are implemented. With custom platforms, you can fully control the infrastructure configuration of each cloud resource based on the needs, policies and constraints of your team.
-
-- Enhanced Security: Some organizations have stringent security requirements. With custom platforms, you can embed additional security checks, audits, or encryption layers to suit these needs.
-
-- Optimizations: Your organization may have developed optimization strategies that can help reduce costs or improve performance when deploying applications. Integrating these strategies into a custom platform can make them a seamless part of the deployment process.
-
-### Creating a custom platform
-
-Developing a custom platform requires understanding and adhering to the `IPlatform` interface. This ensures that your custom platform can be integrated smoothly with the Wing compilation process.
-
-The IPlatform interface is defined as follows:
-
-```ts
-export interface IPlatform {
- // Define the target compatibility of the platform
- readonly target: string;
-
- // Define the App that will be used for creating resources
- newApp?(appProps: AppProps): App;
-
- // Define overrides for concrete resources
- newInstance?(type: string, scope: Construct, id: string, props: any): any;
-
- // Synthesis Hooks
- preSynth?(app: Construct): void;
- postSynth?(config: any): any;
- validate?(config: any): any;
-}
-```
-
-### Using a custom platform
-
-When running the `wing compile` command, the `--platform` option is used to specify the platform provider(s) you wish to use. This option accepts variadic arguments, which means you can specify any number of platforms.
-
-The specified platform can be a built-in platform, or a path to a custom platform. For example, if you have a custom platform named `my-platform`, you can specify it as follows:
-
-```sh
-wing compile --platform tf-aws --platform ../my-platform
-```
-
-### Synthesis Hooks
-
-In the above interface there are three methods that are categorized as synthesis hooks. These hooks are called by the compiler at various points during the compilation process. They allow you to hook into the compilation process and apply customizations.
-
-Your platform only needs to implement the methods that are relevant to your use case. For example, if you are creating a platform that is designed to apply additional security configurations for your organization, then you may only need to implement the `preSynth` hook.
-
-Lets take a look at what each hook is responsible for:
-
-:::info Examples Incoming
-The following examples of hooks use simple JavaScript files for brevity. However, you can and probably would want to build your platform as a Node library to package and distribute it. Examples of this are coming soon.
-:::
-
-### `preSynth` hook
-
-API Reference
-```ts
-preSynth(app: Construct): void;
-```
-
-This hook is called before the compiler begins to synthesize. In the context of a
-Terraform-based platform like `tf-aws`, this hook will have access to the root
-node of the construct tree. This allows the platform to add or change [CDK for
-Terraform](https://github.com/hashicorp/terraform-cdk) constructs before the
-tree is synthesized.
-
-The following example adds a bucket to the root node.
-```js
-const s3_bucket = require("@cdktf/provider-aws/lib/s3-bucket");
-
-exports.Platform = class MyPlatform {
- preSynth(app) {
- // app is the root node of the construct tree
- new s3_bucket.S3Bucket(app, "MyPlatformBucket", {
- bucket: "my-platform-bucket",
- });
- }
-}
-```
-
-### `postSynth` hook
-
-API Reference
-```ts
-postSynth(config: any): any;
-```
-
-This hook runs after artifacts were synthesized. When compiling to a
-Terraform-based platform like `tf-aws`, the hook will have access
-to the raw Terraform JSON configuration, allowing for manipulation of the JSON
-that is written to the compiled output directory.
-
-This hook is useful for adding customizations that can not be applied in the
-context of the preSynth hook. Its worth noting that this is not meant as a
-validation phase since the config is still mutable by subsequent platforms.
-
-The following example manipulates the Terraform configuration to use a S3
-backend. For brevity, the example uses hard coded values.
-```js
-exports.Platform = class MyPlatform {
- postSynth(config) {
- config.terraform.backend = {
- s3: {
- bucket: "my-wing-state-bucket",
- key: "platforms-rock.tfstate",
- region: "us-east-1",
- }
- }
- return config;
- }
-}
-```
-
-### `validate` hook
-
-API Reference
-```ts
-validate(config: any): void;
-```
-
-This hook is called right after the `postSynth` hook and provided the same
-context object. In the context of a Terraform-based platform like `tf-aws`, this
-is the same Terraform JSON configuration. However, does not allow configuration
-to be mutated, which allows platforms to validate the configuration without
-worrying about another platform mutating after the fact.
-
-The following example validates that buckets all have versioning enabled
-and throw an error during compilation if they don't.
-```js
-exports.Platform = class MyPlatform {
- validate(config) {
- for (const bucketEntry of Object.keys(config.resource.aws_s3_bucket)) {
- const bucket = config.resource.aws_s3_bucket[bucketEntry];
- if (!bucket.versioning.enabled) {
- throw new Error(`Bucket ${bucketEntry} does not have versioning enabled`);
- }
- }
- }
-}
-```
-
-### Defining Custom Platform Parameters
-
-In addition to the hooks mentioned above, you can also define custom parameters for your platform. These parameters can be used to pass
-configuration to your platform and can be optional or required.
-
-Parameters are defined using the `parameters` property of the platform. These parameters are expected to be provided in the form of a
-(JSON schema)[https://json-schema.org/].
-
-The following example shoes how to define three parameters `replicateAllBuckets`, `bucketsToReplicate` and `replicaRegion` for a custom platform.
-Our platform's logic will either replicate all buckets if `replicateAllBuckets` is set to `true` or replicate only the buckets specified in `bucketsToReplicate` to the region specified in `replicaRegion`.
-
-Its important to understand the relationship between these parameters, as in if `replicateAllBuckets` is set to `true` then `bucketsToReplicate` is not required.
-Whereas no matter the value of `replicateAllBuckets`, `replicaRegion` is always required.
-
-Luckily, JSON schema allows us to define these relationships and constraints like so:
-
-```js
-class MyPlatform {
- parameters = {
- type: "object",
- required: ["replicateAllBuckets", "replicaRegion"],
- properties: {
- replicateAllBuckets: {
- type: "boolean",
- },
- nameOfBucketsToReplicate: {
- type: "array",
- items: {
- type: "string",
- },
- },
- replicaRegion: {
- type: "string",
- },
- },
- "$comment": "Here in an allOf we can define multiple conditions that must be met for the schema to be valid",
- allOf: [
- {
- if: { properties: { replicateAllBuckets: { const: false }} },
- then: { required: ["nameOfBucketsToReplicate"] },
- }
- ]
- }
-}
-```