add support for outputting flow types - fixes #665

This commit is contained in:
Sebastian McKenzie
2015-03-01 00:32:36 +11:00
parent fb04b2561f
commit e945f0d10f
61 changed files with 965 additions and 98 deletions

View File

@@ -24,7 +24,7 @@ exports.esvalid = function (ast, code, loc) {
if (errors.length) {
var msg = [];
_.each(errors, function (err) {
msg.push(err.message + " - " + JSON.stringify(err.node));
msg.push(err.message + " - " + util.inspect(err.node));
});
throw new Error(loc + ": " + msg.join(". ") + "\n" + code);
}

View File

@@ -97,7 +97,7 @@ var run = function (task, done) {
if (!execCode || actualCode) {
result = transform(actualCode, getOpts(actual));
checkAst(result, actual);
actualCode = result.code;
actualCode = result.code.trim();
try {
chai.expect(actualCode).to.be.equal(expectCode, actual.loc + " !== " + expect.loc);

View File

@@ -0,0 +1,6 @@
var a: number[];
var a: ?number[];
var a: (?number)[];
var a: () => number[];
var a: (() => number)[];
var a: typeof A[];

View File

@@ -0,0 +1,6 @@
var a: number[];
var a: ?number[];
var a: (?number)[];
var a: () => number[];
var a: (() => number)[];
var a: typeof A[];

View File

@@ -0,0 +1,5 @@
var a: { (): number };
var a: { (): number; };
var a: { (): number; y: string; (x: string): string };
var a: { <T>(x: T): number; };
interface A { (): number; };

View File

@@ -0,0 +1,5 @@
var a: { (): number };
var a: { (): number };
var a: { y: string; (): number; (x: string): string };
var a: { <T>(x: T): number };
interface A { (): number };

View File

@@ -0,0 +1,5 @@
declare module A {}
declare module "./a/b.js" {}
declare module A { declare var x: number; }
declare module A { declare function foo(): number; }
declare module A { declare class B { foo(): number; } }

View File

@@ -0,0 +1,11 @@
declare module A {}
declare module "./a/b.js" {}
declare module A {
declare var x: number;
}
declare module A {
declare function foo(): number;
}
declare module A {
declare class B { foo(): number }
}

View File

@@ -0,0 +1,11 @@
declare var foo
declare var foo;
declare function foo(): void
declare function foo(): void;
declare function foo<T>(): void;
declare function foo(x: number, y: string): void;
declare class A {}
declare class A<T> extends B<T> { x: number }
declare class A { static foo(): number; static x : string }
declare class A { static [ indexer: number]: string }
declare class A { static () : number }

View File

@@ -0,0 +1,11 @@
declare var foo;
declare var foo;
declare function foo(): void;
declare function foo(): void;
declare function foo<T>(): void;
declare function foo(x: number, y: string): void;
declare class A {}
declare class A<T> extends B<T> { x: number }
declare class A { static foo(): number; static x: string }
declare class A { static [indexer: number]: string }
declare class A { static (): number }

View File

@@ -0,0 +1,9 @@
interface A {};
interface A extends B {};
interface A<T> extends B<T>, C<T> {};
interface A { foo: () => number; };
interface Dictionary { [index: string]: string; length: number; };
class Foo implements Bar {}
class Foo extends Bar implements Bat, Man<number> {}
class Foo extends class Bar implements Bat {} {}
class Foo extends class Bar implements Bat {} implements Man {}

View File

@@ -0,0 +1,9 @@
interface A {};
interface A extends B {};
interface A<T> extends B<T>, C<T> {};
interface A { foo(): number };
interface Dictionary { length: number; [index: string]: string };
class Foo implements Bar {}
class Foo extends Bar implements Bat, Man<number> {}
class Foo extends class Bar implements Bat {} {}
class Foo extends class Bar implements Bat {} implements Man {}

View File

@@ -0,0 +1,4 @@
var a: A.B;
var a: A.B.C;
var a: A.B<T>;
var a: typeof A.B<T>;

View File

@@ -0,0 +1,4 @@
var a: A.B;
var a: A.B.C;
var a: A.B<T>;
var a: typeof A.B<T>;

View File

@@ -0,0 +1,2 @@
function createElement(tagName: "div"): HTMLDivElement {}
function createElement(tagName: 'div'): HTMLDivElement {}

View File

@@ -0,0 +1,2 @@
function createElement(tagName: "div"): HTMLDivElement {}
function createElement(tagName: "div"): HTMLDivElement {}

View File

@@ -0,0 +1,4 @@
var a: [] = [];
var a: [Foo<T>] = [foo];
var a: [number,] = [123,];
var a: [number, string] = [123, "duck"];

View File

@@ -0,0 +1,4 @@
var a: [] = [];
var a: [Foo<T>] = [foo];
var a: [number] = [123];
var a: [number, string] = [123, "duck"];

View File

@@ -0,0 +1,3 @@
type FBID = number;
type Foo<T> = Bar<T>
export type Foo = number;

View File

@@ -0,0 +1,3 @@
type FBID = number;
type Foo<T> = Bar<T>;
export type Foo = number;

View File

@@ -0,0 +1,97 @@
function foo(numVal: any) {}
function foo(numVal: number) {}
function foo(numVal: number, strVal: string) {}
function foo(numVal: number, untypedVal) {}
function foo(untypedVal, numVal: number) {}
function foo(nullableNum: ?number) {}
function foo(callback: () => void) {}
function foo(callback: () => number) {}
function foo(callback: (_: bool) => number) {}
function foo(callback: (_1: bool, _2: string) => number) {}
function foo(callback: (_1: bool, ...foo: Array<number>) => number) {}
function foo(): number{}
function foo():() => void {}
function foo():(_:bool) => number{}
function foo():(_?:bool) => number{}
function foo(): {} {}
function foo<T>() {}
function foo<T,S>() {}
a = function<T,S>() {};
a = { set fooProp(value: number) {} };
a = { set fooProp(value: number): void {} };
a = { get fooProp():number{} };
a = { id<T>(x: T): T {} };
a = { *id<T>(x: T): T {} };
a = { async id<T>(x: T): T {} };
a = { 123<T>(x: T): T {} };
class Foo {
set fooProp(value: number) {}
}
class Foo {
set fooProp(value: number): void {}
}
class Foo {
get fooProp(): number {}
}
var numVal: number;
var numVal: number = otherNumVal;
var a: { numVal: number };
var a: { numVal: number; };
var a: { numVal: number; [indexer: string]: number };
var a: ?{ numVal: number };
var a: { numVal: number; strVal: string }
var a: { subObj: {strVal: string} }
var a: { subObj: ?{strVal: string} }
var a: { param1: number; param2: string }
var a: { param1: number; param2?: string }
var a: { [a: number]: string; [b: number]: string; };
var a: { add(x: number, ...y: Array<string>): void };
var a: { id<T>(x: T): T; };
var a:Array<number> = [1, 2, 3]
a = class Foo<T> {}
a = class Foo<T> extends Bar<T> {}
class Foo<T> {}
class Foo<T> extends Bar<T> {}
class Foo<T> extends mixin(Bar) {}
class Foo<T> {
bar<U>():number { return 42; }
}
class Foo {
"bar"<T>() {}
}
function foo(requiredParam, optParam?) {}
class Foo {
prop1: string;
prop2: number;
}
class Foo {
static prop1: string;
prop2: number;
}
var x: number | string = 4;
class Array { concat(items:number | string) {}; }
var x: () => number | () => string = fn;
var x: typeof Y = Y;
var x: typeof Y | number = Y;
var {x}: {x: string; } = { x: "hello" };
var {x}: {x: string } = { x: "hello" };
var [x]: Array<string> = [ "hello" ];
function foo({x}: { x: string; }) {}
function foo([x]: Array<string>) {}
function foo(...rest: Array<number>) {}
(function (...rest: Array<number>) {});
((...rest: Array<number>) => rest);
var a: Map<string, Array<string> >
var a: Map<string, Array<string>>
var a: number[]
var a: ?string[]
var a: Promise<bool>[]
var a:(...rest:Array<number>) => number
var identity: <T>(x: T) => T
var identity: <T>(x: T, ...y:T[]) => T
import type foo from "bar";
import type { foo, bar } from "baz";
import type { foo as bar } from "baz";
import type from "foo";
import type, { foo } from "bar";
import type * as namespace from "bar";

View File

@@ -0,0 +1,101 @@
function foo(numVal: any) {}
function foo(numVal: number) {}
function foo(numVal: number, strVal: string) {}
function foo(numVal: number, untypedVal) {}
function foo(untypedVal, numVal: number) {}
function foo(nullableNum: ?number) {}
function foo(callback: () => void) {}
function foo(callback: () => number) {}
function foo(callback: (_: bool) => number) {}
function foo(callback: (_1: bool, _2: string) => number) {}
function foo(callback: (_1: bool, ...foo: Array<number>) => number) {}
function foo(): number {}
function foo(): () => void {}
function foo(): (_: bool) => number {}
function foo(): (_?: bool) => number {}
function foo(): {} {}
function foo<T>() {}
function foo<T, S>() {}
a = function <T, S>() {};
a = { set fooProp(value: number) {} };
a = { set fooProp(value: number): void {} };
a = { get fooProp(): number {} };
a = { id<T>(x: T): T {} };
a = { *id<T>(x: T): T {} };
a = { async id<T>(x: T): T {} };
a = { 123<T>(x: T): T {} };
class Foo {
set fooProp(value: number) {}
}
class Foo {
set fooProp(value: number): void {}
}
class Foo {
get fooProp(): number {}
}
var numVal: number;
var numVal: number = otherNumVal;
var a: { numVal: number };
var a: { numVal: number };
var a: { numVal: number; [indexer: string]: number };
var a: ?{ numVal: number };
var a: { numVal: number; strVal: string };
var a: { subObj: { strVal: string } };
var a: { subObj: ?{ strVal: string } };
var a: { param1: number; param2: string };
var a: { param1: number; param2?: string };
var a: { [a: number]: string; [b: number]: string };
var a: { add(x: number, ...y: Array<string>): void };
var a: { id<T>(x: T): T };
var a: Array<number> = [1, 2, 3];
a = class Foo<T> {};
a = class Foo<T> extends Bar<T> {};
class Foo<T> {}
class Foo<T> extends Bar<T> {}
class Foo<T> extends mixin(Bar) {}
class Foo<T> {
bar<U>(): number {
return 42;
}
}
class Foo {
"bar"<T>() {}
}
function foo(requiredParam, optParam?) {}
class Foo {
prop1: string;
prop2: number;
}
class Foo {
static prop1: string;
prop2: number;
}
var x: number | string = 4;
class Array {
concat(items: number | string) {}
}
var x: () => number | () => string = fn;
var x: typeof Y = Y;
var x: typeof Y | number = Y;
var { x }: { x: string } = { x: "hello" };
var { x }: { x: string } = { x: "hello" };
var [x]: Array<string> = ["hello"];
function foo({ x }: { x: string }) {}
function foo([x]: Array<string>) {}
function foo(...rest: Array<number>) {}
(function (...rest: Array<number>) {});
(...rest: Array<number>) => rest;
var a: Map<string, Array<string>>;
var a: Map<string, Array<string>>;
var a: number[];
var a: ?string[];
var a: Promise<bool>[];
var a: (...rest: Array<number>) => number;
var identity: <T>(x: T) => T;
var identity: <T>(x: T, ...y: T[]) => T;
import type foo from "bar";
import type { foo, bar } from "baz";
import type { foo as bar } from "baz";
import type from "foo";
import type, { foo } from "bar";
import type * as namespace from "bar";

View File

@@ -0,0 +1,4 @@
(xxx: number);
({ xxx: 0, yyy: "hey" }: { xxx: number; yyy: string });
(xxx => xxx + 1: (xxx: number) => number);
((xxx: number), (yyy: string));

View File

@@ -0,0 +1,4 @@
(xxx: number);
({ xxx: 0, yyy: "hey" }: { xxx: number; yyy: string });
(xxx => xxx + 1: (xxx: number) => number);
(xxx: number), (yyy: string);

View File

@@ -0,0 +1,5 @@
{
"experimental": true,
"whitelist": ["flow"],
"noCheckAst": true
}

View File

@@ -0,0 +1,6 @@
var a: number[];
var a: ?number[];
var a: (?number)[];
var a: () => number[];
var a: (() => number)[];
var a: typeof A[];

View File

@@ -0,0 +1,6 @@
var a;
var a;
var a;
var a;
var a;
var a;

View File

@@ -0,0 +1,5 @@
var a: { (): number };
var a: { (): number; };
var a: { (): number; y: string; (x: string): string };
var a: { <T>(x: T): number; };
interface A { (): number; }

View File

@@ -0,0 +1,4 @@
var a;
var a;
var a;
var a;

View File

@@ -0,0 +1,5 @@
declare module A {}
declare module "./a/b.js" {}
declare module A { declare var x: number; }
declare module A { declare function foo(): number; }
declare module A { declare class B { foo(): number; } }

View File

@@ -0,0 +1,11 @@
declare var foo
declare var foo;
declare function foo(): void
declare function foo(): void;
declare function foo<T>(): void;
declare function foo(x: number, y: string): void;
declare class A {}
declare class A<T> extends B<T> { x: number }
declare class A { static foo(): number; static x : string }
declare class A { static [ indexer: number]: string }
declare class A { static () : number }

View File

@@ -0,0 +1,9 @@
interface A {}
interface A extends B {}
interface A<T> extends B<T>, C<T> {}
interface A { foo: () => number; }
interface Dictionary { [index: string]: string; length: number; }
class Foo implements Bar {}
class Foo2 extends Bar implements Bat, Man<number> {}
class Foo3 extends class Bar implements Bat {} {}
class Foo4 extends class Bar implements Bat {} implements Man {}

View File

@@ -0,0 +1,4 @@
class Foo {}
class Foo2 extends Bar {}
class Foo3 extends class Bar {} {}
class Foo4 extends class Bar {} {}

View File

@@ -0,0 +1,4 @@
var a: A.B;
var a: A.B.C;
var a: A.B<T>;
var a: typeof A.B<T>;

View File

@@ -0,0 +1,4 @@
var a;
var a;
var a;
var a;

View File

@@ -0,0 +1,2 @@
function createElement(tagName: "div"): HTMLDivElement {}
function createElement(tagName: 'div'): HTMLDivElement {}

View File

@@ -0,0 +1,2 @@
function createElement(tagName) {}
function createElement(tagName) {}

View File

@@ -0,0 +1,4 @@
var a: [] = [];
var a: [Foo<T>] = [foo];
var a: [number,] = [123,];
var a: [number, string] = [123, "duck"];

View File

@@ -0,0 +1,4 @@
var a = [];
var a = [foo];
var a = [123];
var a = [123, "duck"];

View File

@@ -0,0 +1,3 @@
type FBID = number;
type Foo<T> = Bar<T>
export type Foo = number;

View File

@@ -0,0 +1,97 @@
function foo(numVal: any) {}
function foo(numVal: number) {}
function foo(numVal: number, strVal: string) {}
function foo(numVal: number, untypedVal) {}
function foo(untypedVal, numVal: number) {}
function foo(nullableNum: ?number) {}
function foo(callback: () => void) {}
function foo(callback: () => number) {}
function foo(callback: (_: bool) => number) {}
function foo(callback: (_1: bool, _2: string) => number) {}
function foo(callback: (_1: bool, ...foo: Array<number>) => number) {}
function foo(): number{}
function foo():() => void {}
function foo():(_:bool) => number{}
function foo():(_?:bool) => number{}
function foo(): {} {}
function foo<T>() {}
function foo<T,S>() {}
a = function<T,S>() {};
a = { set fooProp(value: number) {} };
a = { set fooProp(value: number): void {} };
a = { get fooProp():number{} };
a = { id<T>(x: T): T {} };
a = { *id<T>(x: T): T {} };
a = { async id<T>(x: T): T {} };
a = { 123<T>(x: T): T {} };
class Foo {
set fooProp(value: number) {}
}
class Foo2 {
set fooProp(value: number): void {}
}
class Foo3 {
get fooProp(): number {}
}
var numVal: number;
var numVal: number = otherNumVal;
var a: { numVal: number };
var a: { numVal: number; };
var a: { numVal: number; [indexer: string]: number };
var a: ?{ numVal: number };
var a: { numVal: number; strVal: string }
var a: { subObj: {strVal: string} }
var a: { subObj: ?{strVal: string} }
var a: { param1: number; param2: string }
var a: { param1: number; param2?: string }
var a: { [a: number]: string; [b: number]: string; };
var a: { add(x: number, ...y: Array<string>): void };
var a: { id<T>(x: T): T; };
var a:Array<number> = [1, 2, 3]
a = class Foo<T> {}
a = class Foo<T> extends Bar<T> {}
class Foo4<T> {}
class Foo5<T> extends Bar<T> {}
class Foo6<T> extends mixin(Bar) {}
class Foo7<T> {
bar<U>():number { return 42; }
}
class Foo8 {
"bar"<T>() {}
}
function foo(requiredParam, optParam?) {}
class Foo9 {
prop1: string;
prop2: number;
}
class Foo10 {
static prop1: string;
prop2: number;
}
var x: number | string = 4;
class Array { concat(items:number | string) {}; }
var x: () => number | () => string = fn;
var x: typeof Y = Y;
var x: typeof Y | number = Y;
var {x}: {x: string; } = { x: "hello" };
var {x}: {x: string } = { x: "hello" };
var [x]: Array<string> = [ "hello" ];
function foo({x}: { x: string; }) {}
function foo([x]: Array<string>) {}
function foo(...rest: Array<number>) {}
(function (...rest: Array<number>) {});
((...rest: Array<number>) => rest);
var a: Map<string, Array<string> >
var a: Map<string, Array<string>>
var a: number[]
var a: ?string[]
var a: Promise<bool>[]
var a:(...rest:Array<number>) => number
var identity: <T>(x: T) => T
var identity: <T>(x: T, ...y:T[]) => T
import type foo from "bar";
import type { foo2, bar } from "baz";
import type { foo as bar2 } from "baz";
import type from "foo";
import type2, { foo3 } from "bar";
import type * as namespace from "bar";

View File

@@ -0,0 +1,92 @@
function foo(numVal) {}
function foo(numVal) {}
function foo(numVal, strVal) {}
function foo(numVal, untypedVal) {}
function foo(untypedVal, numVal) {}
function foo(nullableNum) {}
function foo(callback) {}
function foo(callback) {}
function foo(callback) {}
function foo(callback) {}
function foo(callback) {}
function foo() {}
function foo() {}
function foo() {}
function foo() {}
function foo() {}
function foo() {}
function foo() {}
a = function () {};
a = { set fooProp(value) {} };
a = { set fooProp(value) {} };
a = { get fooProp() {} };
a = { id(x) {} };
a = { *id(x) {} };
a = { async id(x) {} };
a = { 123(x) {} };
class Foo {
set fooProp(value) {}
}
class Foo2 {
set fooProp(value) {}
}
class Foo3 {
get fooProp() {}
}
var numVal;
var numVal = otherNumVal;
var a;
var a;
var a;
var a;
var a;
var a;
var a;
var a;
var a;
var a;
var a;
var a;
var a = [1, 2, 3];
a = class Foo {};
a = class Foo extends Bar {};
class Foo4 {}
class Foo5 extends Bar {}
class Foo6 extends mixin(Bar) {}
class Foo7 {
bar() {
return 42;
}
}
class Foo8 {
"bar"() {}
}
function foo(requiredParam, optParam?) {}
class Foo9 {}
class Foo10 {}
var x = 4;
class Array {
concat(items) {}
}
var x = fn;
var x = Y;
var x = Y;
var { x } = { x: "hello" };
var { x } = { x: "hello" };
var [x] = ["hello"];
function foo({ x }) {}
function foo([x]) {}
function foo(...rest) {}
(function (...rest) {});
(...rest) => rest;
var a;
var a;
var a;
var a;
var a;
var a;
var identity;
var identity;
import type from "foo";
import type2, { foo3 } from "bar";

View File

@@ -0,0 +1,4 @@
(xxx: number);
({ xxx: 0, yyy: "hey" }: { xxx: number; yyy: string });
(xxx => xxx + 1: (xxx: number) => number);
((xxx: number), (yyy: string));

View File

@@ -0,0 +1,4 @@
xxx;
({ xxx: 0, yyy: "hey" });
xxx => xxx + 1;
xxx, yyy;