Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: typescript and esm in preflight #6157

Merged
merged 23 commits into from
Apr 12, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
17 changes: 11 additions & 6 deletions apps/wingcli-v2/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,6 @@ use lazy_static::lazy_static;
use strum::{Display, EnumString};
use wingc::compile;

pub const VERSION: &str = env!("CARGO_PKG_VERSION");

lazy_static! {
static ref HOME_PATH: PathBuf = home_dir().expect("Could not find home directory");
pub static ref WING_CACHE_DIR: Utf8PathBuf =
Expand Down Expand Up @@ -90,6 +88,10 @@ fn command_build(source_file: Utf8PathBuf, target: Option<Target>) -> Result<(),
install_sdk()?;
} else {
// TODO: check that the SDK version matches the CLI version
if cfg!(test) {
// For now, always reinstall the SDK in tests
install_sdk()?;
}
}
tracing::info!("Using SDK at {}", sdk_root);

Expand Down Expand Up @@ -117,10 +119,13 @@ fn install_sdk() -> Result<(), Box<dyn Error>> {

std::fs::create_dir_all(WING_CACHE_DIR.as_str())?;
let mut install_command = std::process::Command::new("npm");
install_command
.arg("install")
.arg(format!("@winglang/sdk@{VERSION}"))
.arg("esbuild"); // TODO: should this not be an optional dependency?
install_command.arg("install").arg("esbuild"); // TODO: should this not be an optional dependency?
if cfg!(test) {
install_command.arg(format!("file:{}/../../libs/wingsdk", env!("CARGO_MANIFEST_DIR")));
} else {
install_command.arg(format!("@winglang/sdk@{}", env!("CARGO_PKG_VERSION")));
}

install_command.current_dir(WING_CACHE_DIR.as_str());
install_command.stdout(std::process::Stdio::piped());
install_command.stderr(std::process::Stdio::piped());
Expand Down
11 changes: 3 additions & 8 deletions docs/docs/03-language-reference.md
Original file line number Diff line number Diff line change
Expand Up @@ -1885,7 +1885,8 @@ Mapping JSII types to Wing types:

## 5.2 JavaScript

The `extern "<commonjs module path>"` modifier can be used on method declarations in classes to indicate that a method is backed by an implementation imported from a JavaScript module. The module must be a relative path and will be loaded via [require()](https://nodejs.org/api/modules.html#requireid).
The `extern "<javascript module path>"` modifier can be used on method declarations in classes to indicate that a method is backed by an implementation imported from a JavaScript module. The module must be a relative path and will be loaded via [require()](https://nodejs.org/api/modules.html#requireid).
This module can be either CJS or ESM and may be written in JavaScript or TypeScript.

In the following example, the static inflight method `makeId` is implemented
in `helper.js`:
Expand Down Expand Up @@ -1920,13 +1921,7 @@ matching name (without any case conversion).

Extern methods do not support access to class's members through `this`, so they must be declared `static`.

### 5.2.1 TypeScript

It is possible to use TypeScript to write helpers, but at the moment this is not
directly supported by Wing. This means that you will need to setup the TypeScript toolchain
to compile your code to JavaScript and then use `extern` against the JavaScript file.

### 5.2.2 Type model
### 5.2.1 Type model

The table below shows the mapping between Wing types and JavaScript values, shown with TypeScript types.
When calling **extern** function, the parameter and return types are **assumed** to be satisfied by the called function.
Expand Down
4 changes: 2 additions & 2 deletions docs/docs/07-examples/10-using-javascript.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ keywords: [example, javascript, extern, typescript, js, ts]

Calling a Javascript function from Wing requires two steps.

1. Create a .js file that exports some functions
1. Create a .js/.ts file that exports some functions

```js
// util.js
Expand All @@ -16,7 +16,7 @@ exports.isValidUrl = function (url) {
};
```

In preflight, this file must be a CommonJS module written in Javascript. Inflight, it may be CJS/ESM and either JavaScript or TypeScript.
It may be CJS/ESM written in either JavaScript or TypeScript.

2. Use the `extern` keyword in a class to expose the function to Wing. Note that this must be `static`. It may also be `inflight`

Expand Down
2 changes: 1 addition & 1 deletion examples/tests/invalid/extern_static.test.w
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
class Foo {
extern "../valid/external_js.js" inflight getGreeting(name: str): str;
extern "../valid/external_ts.ts" inflight getGreeting(name: str): str;
//^ Error: extern methods must be declared "static"
}
2 changes: 1 addition & 1 deletion examples/tests/invalid/lib/extern_above.w
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
class Foo1 {
extern "../../valid/external_js.js" static getGreeting(name: str): str;
extern "../../valid/external_ts.ts" static getGreeting(name: str): str;
//^ must be a sub directory of the entrypoint
}
6 changes: 6 additions & 0 deletions examples/tests/valid/bring_local.test.w
Original file line number Diff line number Diff line change
Expand Up @@ -5,15 +5,21 @@ bring "./store.w" as file1;
bring "./subdir/subfile.w" as file2;
bring "./subdir/empty.w" as file3;
bring math;
bring expect;

// classes from other files can be used
let store = new file1.Store();
let q = new file2.Q();
expect.equal(file2.Q.preflightGreet("foo"), "Hello foo");

test "add data to store" {
store.store("foo");
}

test "greet" {
expect.equal(file2.Q.greet("bar"), "Hello bar");
}

// structs from other files can be used
let s = file1.Point {
x: 1,
Expand Down
20 changes: 12 additions & 8 deletions examples/tests/valid/extern_implementation.test.w
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
bring cloud;

class Foo {
extern "./external_js.js" pub static getGreeting(name: str): str;
extern "./external_js.js" static inflight regexInflight(pattern: str, text: str): bool;
extern "./external_js.js" static inflight getUuid(): str;
extern "./external_js.js" static inflight getData(): str;
extern "./external_js.js" pub static inflight print(msg: str): void;
extern "./external_js.js" pub static preflightBucket(bucket: cloud.Bucket, id: str): Json;
extern "./external_ts.ts" pub static getGreeting(name: str): str;
extern "./external_ts.ts" static inflight regexInflight(pattern: str, text: str): bool;
extern "./external_ts.ts" static inflight getUuid(): str;
extern "./external_ts.ts" static inflight getData(): str;
extern "./external_ts.ts" pub static inflight print(msg: str): void;
extern "./external_ts.ts" pub static preflightBucket(bucket: cloud.Bucket, id: str): void;
MarkMcCulloh marked this conversation as resolved.
Show resolved Hide resolved

pub inflight call() {
assert(Foo.regexInflight("[a-z]+-\\d+", "abc-123"));
Expand All @@ -22,10 +22,14 @@ assert(Foo.getGreeting("Wingding") == "Hello, Wingding!");
let f = new Foo();

let bucket = new cloud.Bucket() as "my-bucket";
let result = Foo.preflightBucket(bucket, "my-bucket");
MarkMcCulloh marked this conversation as resolved.
Show resolved Hide resolved
Foo.preflightBucket(bucket, "my-bucket");

test "call" {
let func = new cloud.Function(inflight () => {
f.call();
});

test "call" {
func.invoke();
}

test "console" {
Expand Down
25 changes: 0 additions & 25 deletions examples/tests/valid/external_js.js

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ export default interface extern {
getData: () => Promise<string>,
getGreeting: (name: string) => string,
getUuid: () => Promise<string>,
preflightBucket: (bucket: Bucket, id: string) => Readonly<any>,
preflightBucket: (bucket: Bucket, id: string) => void,
print: (msg: string) => Promise<void>,
regexInflight: (pattern: string, text: string) => Promise<boolean>,
}
Expand Down
31 changes: 31 additions & 0 deletions examples/tests/valid/external_ts.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import assert from "assert";
import extern from "./external_ts.extern";

export const getGreeting: extern["getGreeting"] = function (name) {
return `Hello, ${name}!`;
};

export const regexInflight: extern["regexInflight"] = async function (
pattern,
text
) {
const regex = new RegExp(pattern);
return regex.test(text);
};

export const getUuid: extern["getUuid"] = async function () {
let uuid = require("uuid");
return uuid.v4();
};

export const getData: extern["getData"] = async function () {
return require("./exported_data.js");
};

export const print: extern["print"] = async function (msg) {
console.log(`printing ${msg}`);
};

export const preflightBucket: extern["preflightBucket"] = (bucket, id) => {
assert.strictEqual(bucket.node.id, id);
};
3 changes: 2 additions & 1 deletion examples/tests/valid/subdir/subfile.w
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
bring math;

pub class Q {
extern "./util.ts" static inflight greet(name: str): str;
pub extern "./util.ts" static inflight greet(name: str): str;
pub extern "./util.ts" static preflightGreet(name: str): str;
}
1 change: 1 addition & 0 deletions examples/tests/valid/subdir/util.extern.d.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
export default interface extern {
greet: (name: string) => Promise<string>,
preflightGreet: (name: string) => string,
}
4 changes: 4 additions & 0 deletions examples/tests/valid/subdir/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,8 @@ import type extern from "./util.extern";

export const greet: extern["greet"] = async (name) => {
return "Hello " + name;
}

export const preflightGreet: extern["preflightGreet"] = (name) => {
return "Hello " + name;
}
2 changes: 2 additions & 0 deletions libs/wingc/src/dtsify/snapshots/declarations.snap
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,7 @@ module.exports = function({ }) {
const $stdlib = require('@winglang/sdk');
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
module.exports = {
...require("./preflight.lib-1.cjs"),
};
Expand All @@ -98,6 +99,7 @@ export * from "./preflight.lib-1.cjs"
const $stdlib = require('@winglang/sdk');
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class ParentClass extends $stdlib.std.Resource {
constructor($scope, $id, ) {
super($scope, $id);
Expand Down
2 changes: 2 additions & 0 deletions libs/wingc/src/dtsify/snapshots/optionals.snap
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ module.exports = function({ }) {
const $stdlib = require('@winglang/sdk');
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
module.exports = {
...require("./preflight.lib-1.cjs"),
};
Expand All @@ -64,6 +65,7 @@ export * from "./preflight.lib-1.cjs"
const $stdlib = require('@winglang/sdk');
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class ParentClass extends $stdlib.std.Resource {
constructor($scope, $id, ) {
super($scope, $id);
Expand Down
14 changes: 12 additions & 2 deletions libs/wingc/src/jsify.rs
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@ const ENV_WING_IS_TEST: &str = "$wing_is_test";
const OUTDIR_VAR: &str = "$outdir";
const PLATFORMS_VAR: &str = "$platforms";
const HELPERS_VAR: &str = "$helpers";
const EXTERN_VAR: &str = "$extern";

const ROOT_CLASS: &str = "$Root";
const JS_CONSTRUCTOR: &str = "constructor";
Expand Down Expand Up @@ -170,7 +171,7 @@ impl<'a> JSifier<'a> {

output.line("\"use strict\";");

output.line(format!("const {} = require('{}');", STDLIB, STDLIB_MODULE));
output.line(format!("const {STDLIB} = require('{STDLIB_MODULE}');"));

if is_entrypoint {
output.line(format!(
Expand All @@ -187,6 +188,9 @@ impl<'a> JSifier<'a> {
// "std" is implicitly imported
output.line(format!("const std = {STDLIB}.{WINGSDK_STD_MODULE};"));
output.line(format!("const {HELPERS_VAR} = {STDLIB}.helpers;"));
output.line(format!(
"const {EXTERN_VAR} = {HELPERS_VAR}.createExternRequire(__dirname);"
));
output.add_code(imports);

if is_entrypoint {
Expand Down Expand Up @@ -1452,9 +1456,15 @@ impl<'a> JSifier<'a> {
format!("{up_dirs}{rel_path}")
};

let require = if ctx.visit_ctx.current_phase() == Phase::Inflight {
"require"
} else {
EXTERN_VAR
};

new_code!(
&func_def.span,
format!("return (require(\"{require_path}\")[\"{name}\"])("),
format!("return ({require}(\"{require_path}\")[\"{name}\"])("),
parameters.clone(),
")"
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
const cloud = $stdlib.cloud;
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
const cloud = $stdlib.cloud;
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ const $outdir = process.env.WING_SYNTH_DIR ?? ".";
const $wing_is_test = process.env.WING_IS_TEST === "true";
const std = $stdlib.std;
const $helpers = $stdlib.helpers;
const $extern = $helpers.createExternRequire(__dirname);
class $Root extends $stdlib.std.Resource {
constructor($scope, $id) {
super($scope, $id);
Expand Down
Loading
Loading