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(rosetta): find fixtures based on submodules #3131

Merged
merged 7 commits into from
Nov 8, 2021
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
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
89 changes: 78 additions & 11 deletions packages/jsii-rosetta/lib/fixtures.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import * as fs from 'fs-extra';
import * as path from 'path';
import { createSourceFile, ScriptKind, ScriptTarget, SyntaxKind } from 'typescript';

import { TypeScriptSnippet, SnippetParameters } from './snippet';
import { TypeScriptSnippet, SnippetParameters, ApiLocation } from './snippet';

/**
* Complete snippets with fixtures, if required
Expand Down Expand Up @@ -31,10 +31,10 @@ export function fixturize(snippet: TypeScriptSnippet, loose = false): TypeScript
parameters[SnippetParameters.$COMPILATION_DIRECTORY] = path.join(directory, path.dirname(literateSource));
} else if (parameters[SnippetParameters.FIXTURE]) {
// Explicitly requested fixture must exist, unless we are operating in loose mode
source = loadAndSubFixture(directory, parameters.fixture, source, !loose);
source = loadAndSubFixture(directory, snippet.location.api, parameters.fixture, source, !loose);
} else if (parameters[SnippetParameters.NO_FIXTURE] === undefined) {
// Don't explicitly request no fixture
source = loadAndSubFixture(directory, 'default', source, false);
// Don't explicitly request no fixture, load the default.
source = loadAndSubFixture(directory, snippet.location.api, 'default', source, false);
}

return {
Expand All @@ -54,15 +54,43 @@ function loadLiterateSource(directory: string, literateFileName: string) {
return fs.readFileSync(fullPath, { encoding: 'utf-8' });
}

function loadAndSubFixture(directory: string, fixtureName: string, source: string, mustExist: boolean) {
const fixtureFileName = path.join(directory, `rosetta/${fixtureName}.ts-fixture`);
const exists = fs.existsSync(fixtureFileName);
if (!exists && mustExist) {
throw new Error(`Sample uses fixture ${fixtureName}, but not found: ${fixtureFileName}`);
}
if (!exists) {
/**
* Load the fixture with the given name, and substitute the source into it
*
* If no fixture could be found and `mustExist` is true, and error will be thrown.
*
* In principle, the fixture we're looking for is `rosetta/FIXTURE.ts-fixture`.
* However, we want to support an automatic transform of many small packages
* combined into a single large package, perhaps into submodules (i.e., we want
* to support monocdk), and in those cases the names of fixtures might conflict.
* For example, all of them will have a `default.ts-fixture`, and there won't be
* any explicit reference to that file anywhere... yet in the combined
* monopackage we have to distinguish those fixtures.
*
* Therefore, we will consider submodule names as subdirectories, based on the
* API location of the snippet we're fixturizing.
*
* (For example, the fixtures for a type called `monocdk.aws_s3.Bucket` will be
* searched both in `rosetta/aws_s3/default.ts-fixture` as well as
* `rosetta/default.ts-fixture`).
*/
function loadAndSubFixture(
directory: string,
location: ApiLocation,
fixtureName: string,
source: string,
mustExist: boolean,
) {
const candidates = fixtureCandidates(directory, fixtureName, location);
const fixtureFileName = candidates.find((n) => fs.existsSync(n));

if (!fixtureFileName) {
if (mustExist) {
throw new Error(`Sample uses fixture ${fixtureName}, but not found: ${JSON.stringify(candidates)}`);
}
return source;
}

const fixtureContents = fs.readFileSync(fixtureFileName, {
encoding: 'utf-8',
});
Expand Down Expand Up @@ -99,6 +127,45 @@ function loadAndSubFixture(directory: string, fixtureName: string, source: strin
: result;
}

function fixtureCandidates(directory: string, fixtureName: string, location: ApiLocation): string[] {
const ret = new Array<string>();
const fileName = `${fixtureName}.ts-fixture`;
const mods = submodules(location);

ret.push(path.join(directory, 'rosetta', fileName));
for (let i = 0; i < mods.length; i++) {
ret.push(path.join(directory, 'rosetta', ...mods.slice(0, i + 1), fileName));
}

// Most specific one up front
ret.reverse();
return ret;
}

/**
* Return the submodule parts from a given ApiLocation
*/
function submodules(location: ApiLocation): string[] {
switch (location.api) {
case 'file':
return [];
case 'initializer':
case 'member':
case 'type':
case 'parameter':
return middle(location.fqn.split('.'));
case 'moduleReadme':
return location.moduleFqn.split('.').slice(1);
}

function middle(xs: string[]) {
if (xs.length <= 2) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[minor] Why is this a special case?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're right, it doesn't need to be.

return [];
}
return xs.slice(1, xs.length - 1);
}
}

/**
* When embedding code fragments in a fixture, "import" statements must be
* hoisted up to the top of the resulting document, as TypeScript only allows
Expand Down
22 changes: 11 additions & 11 deletions packages/jsii-rosetta/test/commands/extract.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import * as path from 'path';
import { LanguageTablet } from '../../lib';
import * as extract from '../../lib/commands/extract';
import { TARGET_LANGUAGES } from '../../lib/languages';
import { AssemblyFixture, DUMMY_ASSEMBLY_TARGETS } from '../testutil';
import { TestJsiiModule, DUMMY_ASSEMBLY_TARGETS } from '../testutil';

const DUMMY_README = `
Here is an example of how to use ClassA:
Expand All @@ -20,10 +20,10 @@ const defaultExtractOptions = {
validateAssemblies: false,
};

let assembly: AssemblyFixture;
let assembly: TestJsiiModule;
beforeAll(async () => {
// Create an assembly in a temp directory
assembly = await AssemblyFixture.fromSource(
assembly = await TestJsiiModule.fromSource(
{
'index.ts': `
export class ClassA {
Expand All @@ -43,8 +43,8 @@ beforeAll(async () => {
afterAll(async () => assembly.cleanup());

test('extract samples from test assembly', async () => {
const outputFile = path.join(assembly.directory, 'test.tabl.json');
await extract.extractSnippets([assembly.directory], {
const outputFile = path.join(assembly.moduleDirectory, 'test.tabl.json');
await extract.extractSnippets([assembly.moduleDirectory], {
outputFile,
...defaultExtractOptions,
});
Expand All @@ -58,8 +58,8 @@ test('extract samples from test assembly', async () => {
describe('with cache file', () => {
let cacheTabletFile: string;
beforeAll(async () => {
cacheTabletFile = path.join(assembly.directory, 'cache.tabl.json');
await extract.extractSnippets([assembly.directory], {
cacheTabletFile = path.join(assembly.moduleDirectory, 'cache.tabl.json');
await extract.extractSnippets([assembly.moduleDirectory], {
outputFile: cacheTabletFile,
...defaultExtractOptions,
});
Expand All @@ -68,8 +68,8 @@ describe('with cache file', () => {
test('translation does not happen if it can be read from cache', async () => {
const translationFunction = jest.fn().mockResolvedValue({ diagnostics: [], translatedSnippets: [] });

await extract.extractSnippets([assembly.directory], {
outputFile: path.join(assembly.directory, 'dummy.tabl.json'),
await extract.extractSnippets([assembly.moduleDirectory], {
outputFile: path.join(assembly.moduleDirectory, 'dummy.tabl.json'),
cacheTabletFile,
translationFunction,
...defaultExtractOptions,
Expand All @@ -84,8 +84,8 @@ describe('with cache file', () => {
const oldJavaVersion = TARGET_LANGUAGES.java.version;
(TARGET_LANGUAGES.java as any).version = '999';
try {
await extract.extractSnippets([assembly.directory], {
outputFile: path.join(assembly.directory, 'dummy.tabl.json'),
await extract.extractSnippets([assembly.moduleDirectory], {
outputFile: path.join(assembly.moduleDirectory, 'dummy.tabl.json'),
cacheTabletFile,
translationFunction,
...defaultExtractOptions,
Expand Down
20 changes: 10 additions & 10 deletions packages/jsii-rosetta/test/commands/infuse.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import * as path from 'path';
import { extractSnippets } from '../../lib/commands/extract';
import { infuse, DEFAULT_INFUSION_RESULTS_NAME } from '../../lib/commands/infuse';
import { loadAssemblies } from '../../lib/jsii/assemblies';
import { AssemblyFixture, DUMMY_ASSEMBLY_TARGETS } from '../testutil';
import { TestJsiiModule, DUMMY_ASSEMBLY_TARGETS } from '../testutil';

const DUMMY_README = `
Here is an example of how to use ClassA:
Expand All @@ -18,10 +18,10 @@ const DUMMY_README = `

const TABLET_FILE = 'text.tabl.json';

let assembly: AssemblyFixture;
let assembly: TestJsiiModule;
beforeAll(async () => {
// Create an assembly in a temp directory
assembly = await AssemblyFixture.fromSource(
assembly = await TestJsiiModule.fromSource(
{
'index.ts': `
export class ClassA {
Expand All @@ -45,8 +45,8 @@ beforeAll(async () => {
);

// Create a tabletFile in the same directory
await extractSnippets([assembly.directory], {
outputFile: path.join(assembly.directory, TABLET_FILE),
await extractSnippets([assembly.moduleDirectory], {
outputFile: path.join(assembly.moduleDirectory, TABLET_FILE),
includeCompilerDiagnostics: false,
validateAssemblies: false,
});
Expand All @@ -55,22 +55,22 @@ beforeAll(async () => {
afterAll(async () => assembly.cleanup());

test('examples are added in the assembly', async () => {
await infuse([assembly.directory], path.join(assembly.directory, TABLET_FILE));
await infuse([assembly.moduleDirectory], path.join(assembly.moduleDirectory, TABLET_FILE));

const assemblies = await loadAssemblies([assembly.directory], false);
const assemblies = await loadAssemblies([assembly.moduleDirectory], false);
const types = assemblies[0].assembly.types;
expect(types).toBeDefined();
expect(types!['my_assembly.ClassA'].docs?.example).toBeDefined();
});

test('can log to output file', async () => {
await infuse([assembly.directory], path.join(assembly.directory, TABLET_FILE), {
await infuse([assembly.moduleDirectory], path.join(assembly.moduleDirectory, TABLET_FILE), {
log: true,
outputFile: path.join(assembly.directory, DEFAULT_INFUSION_RESULTS_NAME),
outputFile: path.join(assembly.moduleDirectory, DEFAULT_INFUSION_RESULTS_NAME),
});

// assert that the output file exists and there is some information in the file.
const stats = await fs.stat(path.join(assembly.directory, DEFAULT_INFUSION_RESULTS_NAME));
const stats = await fs.stat(path.join(assembly.moduleDirectory, DEFAULT_INFUSION_RESULTS_NAME));

expect(stats.isFile()).toBeTruthy();
expect(stats.size).toBeGreaterThan(0);
Expand Down
39 changes: 39 additions & 0 deletions packages/jsii-rosetta/test/jsii/assemblies.test.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
import * as spec from '@jsii/spec';
import * as fs from 'fs-extra';
import * as mockfs from 'mock-fs';
import * as path from 'path';

import { allTypeScriptSnippets } from '../../lib/jsii/assemblies';
import { SnippetParameters } from '../../lib/snippet';
import { TestJsiiModule, DUMMY_ASSEMBLY_TARGETS } from '../testutil';
import { fakeAssembly } from './fake-assembly';

test('Extract snippet from README', () => {
Expand Down Expand Up @@ -230,3 +232,40 @@ test('Backwards compatibility with literate integ tests', () => {
mockfs.restore();
}
});

test('rosetta fixture from submodule is preferred if it exists', async () => {
const jsiiModule = await TestJsiiModule.fromSource(
{
'index.ts': 'export * as submodule from "./submodule"',
'submodule.ts': `
/**
* @example new ClassA();
*/
export class ClassA {
public someMethod() {
}
}`,
},
{
name: 'my_assembly',
jsii: DUMMY_ASSEMBLY_TARGETS,
},
);
try {
await fs.mkdirp(path.join(jsiiModule.moduleDirectory, 'rosetta', 'submodule'));
await fs.writeFile(
path.join(jsiiModule.moduleDirectory, 'rosetta', 'submodule', 'default.ts-fixture'),
'pick me\n/// here',
);
await fs.writeFile(
path.join(jsiiModule.moduleDirectory, 'rosetta', 'default.ts-fixture'),
'dont pick me\n/// here',
);

const snippets = allTypeScriptSnippets([{ assembly: jsiiModule.assembly, directory: jsiiModule.moduleDirectory }]);

expect(snippets[0].completeSource).toMatch(/^pick me/);
} finally {
await jsiiModule.cleanup();
}
});
8 changes: 4 additions & 4 deletions packages/jsii-rosetta/test/record-references.test.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import { AssemblyFixture, DUMMY_ASSEMBLY_TARGETS } from './testutil';
import { TestJsiiModule, DUMMY_ASSEMBLY_TARGETS } from './testutil';

let assembly: AssemblyFixture;
let assembly: TestJsiiModule;
beforeAll(async () => {
assembly = await AssemblyFixture.fromSource(
assembly = await TestJsiiModule.fromSource(
`
export class ClassA {
public someMethod() {
Expand All @@ -25,7 +25,7 @@ beforeAll(async () => {
);
});

afterAll(async () => assembly.cleanup());
afterAll(() => assembly.cleanup());

test('detect class instantiations', () => {
const translator = assembly.successfullyCompile(`
Expand Down
6 changes: 3 additions & 3 deletions packages/jsii-rosetta/test/syntax-counter.test.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import { AssemblyFixture, DUMMY_ASSEMBLY_TARGETS } from './testutil';
import { TestJsiiModule, DUMMY_ASSEMBLY_TARGETS } from './testutil';

let assembly: AssemblyFixture;
let assembly: TestJsiiModule;
beforeAll(async () => {
assembly = await AssemblyFixture.fromSource(
assembly = await TestJsiiModule.fromSource(
`
export class ClassA {
public someMethod() {
Expand Down
18 changes: 13 additions & 5 deletions packages/jsii-rosetta/test/testutil.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import * as spec from '@jsii/spec';
import * as fs from 'fs-extra';
import { PackageInfo, compileJsiiForTest } from 'jsii';
import * as os from 'os';
Expand All @@ -13,7 +14,10 @@ import {

export type MultipleSources = { [key: string]: string; 'index.ts': string };

export class AssemblyFixture {
/**
* Compile a jsii module from source, and produce an environment in which it is available as a module
*/
export class TestJsiiModule {
public static async fromSource(
source: string | MultipleSources,
packageInfo: Partial<PackageInfo> & { name: string },
Expand Down Expand Up @@ -43,18 +47,22 @@ export class AssemblyFixture {
await fs.writeFile(path.join(modDir, fileName), fileContents);
}

return new AssemblyFixture(modDir);
return new TestJsiiModule(assembly, modDir, tmpDir);
}

private constructor(public readonly directory: string) {}
private constructor(
public readonly assembly: spec.Assembly,
public readonly moduleDirectory: string,
public readonly workspaceDirectory: string,
) {}

/**
* Make a snippet translator for the given source w.r.t this compiled assembly
*/
public successfullyCompile(source: string) {
const location = testSnippetLocation('testutil');
const snippet = typeScriptSnippetFromSource(source, location, false, {
[SnippetParameters.$COMPILATION_DIRECTORY]: this.directory,
[SnippetParameters.$COMPILATION_DIRECTORY]: this.workspaceDirectory,
});
const ret = new SnippetTranslator(snippet, {
includeCompilerDiagnostics: true,
Expand All @@ -69,7 +77,7 @@ export class AssemblyFixture {
}

public async cleanup() {
await fs.remove(this.directory);
await fs.remove(this.moduleDirectory);
}
}

Expand Down