diff --git a/.changes/unreleased/INTERNAL-20241105-111825.yaml b/.changes/unreleased/INTERNAL-20241105-111825.yaml new file mode 100644 index 000000000..70c22d637 --- /dev/null +++ b/.changes/unreleased/INTERNAL-20241105-111825.yaml @@ -0,0 +1,6 @@ +kind: INTERNAL +body: Migrate to redhat-ext-tester +time: 2024-11-05T11:18:25.988501-05:00 +custom: + Issue: "1873" + Repository: vscode-terraform diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index dcc3f4c81..23a33768f 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -50,6 +50,8 @@ jobs: - ubuntu-latest runs-on: ${{ matrix.os }} timeout-minutes: 10 + env: + VSCODE_VERSION: ${{ matrix.vscode }} steps: - name: Checkout Repo uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 @@ -72,3 +74,9 @@ jobs: - name: Run Tests run: npm test if: runner.os != 'Linux' + - name: Run UI Tests + run: xvfb-run --auto-servernum --server-args='-screen 0 1920x1080x24' npm run test:ui + if: runner.os == 'Linux' + - name: Run UI Tests + run: npm run test:ui + if: runner.os != 'Linux' diff --git a/.gitignore b/.gitignore index 7c1f0c3da..828e1609e 100644 --- a/.gitignore +++ b/.gitignore @@ -2,6 +2,7 @@ .terraform .vscode-test .vscode-test-web +.test-storage .test-extensions *.vsix bin diff --git a/.mocharc.js b/.mocharc.js deleted file mode 100644 index a66b8cc19..000000000 --- a/.mocharc.js +++ /dev/null @@ -1,4 +0,0 @@ -// increase default test case timeout to 5 seconds -module.exports = { - timeout: 15000, -}; diff --git a/.prettierignore b/.prettierignore index b3487cba5..95435b7c4 100644 --- a/.prettierignore +++ b/.prettierignore @@ -2,6 +2,7 @@ .vscode-test .vscode-test-web .wdio-vscode-service +.test-storage language-configuration.json node_modules dist diff --git a/.vscode/launch.json b/.vscode/launch.json index 60cc00150..6ddc02479 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -25,6 +25,32 @@ "args": ["--extensionDevelopmentPath=${workspaceFolder}", "--extensionDevelopmentKind=web"], "outFiles": ["${workspaceFolder}/dist/web/**/*.js"], "preLaunchTask": "npm: watch" + }, + { + "name": "Debug UI Tests", + "type": "node", + "request": "launch", + "program": "${workspaceFolder}/node_modules/.bin/extest", + "windows": { + "program": "${workspaceFolder}/node_modules/vscode-extension-tester/out/cli.js" + }, + "args": [ + "setup-and-run", + "${workspaceFolder}/out/test/e2e/specs/**/*.e2e.js", + "--code_settings", + "${workspaceFolder}/src/test/e2e/settings.json", + "--extensions_dir", + ".test-extensions", + "--mocha_config", + "${workspaceFolder}/src/test/e2e/.mocharc.js", + "--uninstall_extension", + "--log_level", + "debug", + "--storage", + ".test-storage" + ], + "console": "integratedTerminal", + "internalConsoleOptions": "neverOpen" } ] } diff --git a/.vscodeignore b/.vscodeignore index d30782373..a170d3ca4 100644 --- a/.vscodeignore +++ b/.vscodeignore @@ -1,3 +1,7 @@ +.mocharc.js +esbuild.mjs +eslint.config.mjs +uitest.mjs .changes .changie.yaml .copywrite.hcl @@ -16,6 +20,7 @@ .vscode-test.mjs .vscodeignore .wdio-vscode-service +.test-storage **/__mocks__ build/ DEVELOPMENT.md @@ -26,3 +31,4 @@ src/ out/ esbuild.js tsconfig.json +.test-extensions diff --git a/eslint.config.mjs b/eslint.config.mjs index ce73ded0f..6eb305d51 100644 --- a/eslint.config.mjs +++ b/eslint.config.mjs @@ -37,9 +37,10 @@ export default [ semi: 'warn', '@typescript-eslint/no-explicit-any': ['warn', { ignoreRestArgs: true }], '@typescript-eslint/naming-convention': 'off', - '@typescript-eslint/no-unsafe-assignment': 'warn', - '@typescript-eslint/no-unsafe-argument': 'warn', - '@typescript-eslint/no-unsafe-member-access': 'warn', + '@typescript-eslint/no-unsafe-assignment': 'off', + '@typescript-eslint/no-unsafe-argument': 'off', + '@typescript-eslint/no-unsafe-member-access': 'off', + '@typescript-eslint/no-unsafe-call': 'off', '@typescript-eslint/restrict-template-expressions': [ 'error', { @@ -64,6 +65,7 @@ export default [ ignores: [ '.vscode-test', '.wdio-vscode-service', + '.test-storage', 'dist', 'out', 'src/test', diff --git a/package-lock.json b/package-lock.json index 9a29ef817..b5bf5e714 100644 --- a/package-lock.json +++ b/package-lock.json @@ -17,7 +17,8 @@ "vscode-languageclient": "^9.0.1", "vscode-uri": "^3.0.7", "which": "^3.0.1", - "zod": "^3.21.4" + "zod": "^3.21.4", + "zod-fixture": "^2.5.2" }, "devDependencies": { "@esbuild-plugins/node-globals-polyfill": "^0.2.3", @@ -3538,6 +3539,15 @@ "url": "https://github.com/fb55/domutils?sponsor=1" } }, + "node_modules/drange": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/drange/-/drange-1.1.1.tgz", + "integrity": "sha512-pYxfDYpued//QpnLIm4Avk7rsNtAtQkUES2cwAYSvD/wd2pKD71gN2Ebj3e7klzXwjocvE8c5vx/1fxwpqmSxA==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, "node_modules/duplexer2": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.1.4.tgz", @@ -7655,6 +7665,19 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/randexp": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/randexp/-/randexp-0.5.3.tgz", + "integrity": "sha512-U+5l2KrcMNOUPYvazA3h5ekF80FHTUG+87SEAmHZmolh1M+i/WyTCxVzmi+tidIa1tM4BSe8g2Y/D3loWDjj+w==", + "license": "MIT", + "dependencies": { + "drange": "^1.0.2", + "ret": "^0.2.0" + }, + "engines": { + "node": ">=4" + } + }, "node_modules/randombytes": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", @@ -7869,6 +7892,15 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/ret": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/ret/-/ret-0.2.2.tgz", + "integrity": "sha512-M0b3YWQs7R3Z917WRQy1HHA7Ba7D8hvZg6UE5mLykJxQVE2ju0IXbGlaHPPlkY+WN7wFP+wUMXmBFA0aV6vYGQ==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, "node_modules/reusify": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", @@ -9921,6 +9953,18 @@ "funding": { "url": "https://github.com/sponsors/colinhacks" } + }, + "node_modules/zod-fixture": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/zod-fixture/-/zod-fixture-2.5.2.tgz", + "integrity": "sha512-JipX1OVrKA3QSdx/k29sK9zocFM8mvdFHu8Wt0htrJ5ZlE8vcmnNfT9iKXMjjc3Hqy8btWtPyP4e03D0IxpVxg==", + "license": "MIT", + "dependencies": { + "randexp": "^0.5.3" + }, + "peerDependencies": { + "zod": ">=3.0.0" + } } } } diff --git a/package.json b/package.json index c36b2fc2a..00f717cd9 100644 --- a/package.json +++ b/package.json @@ -550,6 +550,11 @@ "title": "HCP Terraform: Login", "enablement": "terraform.cloud.signed-in === false" }, + { + "command": "terraform.cloud.logout", + "title": "HCP Terraform: Logout", + "enablement": "terraform.cloud.signed-in === true" + }, { "command": "terraform.cloud.workspaces.refresh", "title": "Refresh", @@ -656,6 +661,10 @@ "command": "terraform.cloud.login", "when": "terraform.cloud.signed-in === false && terraform.cloud.views.visible" }, + { + "command": "terraform.cloud.logout", + "when": "terraform.cloud.signed-in === true && terraform.cloud.views.visible" + }, { "command": "terraform.cloud.organization.picker", "when": "terraform.cloud.signed-in" @@ -928,7 +937,7 @@ ] }, "scripts": { - "prepare": "npm run download:artifacts && cd src/test/e2e && npm install", + "prepare": "npm run download:artifacts", "compile": "npm run check-types && npm run lint && node esbuild.mjs", "compile:prod": "npm run check-types && npm run lint && node esbuild.mjs --production", "compile:tests": "tsc -p .", @@ -947,7 +956,7 @@ "package": "vsce package", "pretest": "npm run compile:tests && npm run compile && npm run lint", "test": "vscode-test", - "test:ui": "npm run compile:tests && extest setup-and-run './out/test/e2e/specs/**/*.e2e.js' --code_version max --extensions_dir .test-extensions", + "test:ui": "node uitest.mjs", "lint": "eslint", "format": "prettier --write .", "check-types": "tsc --noEmit", @@ -963,7 +972,8 @@ "vscode-languageclient": "^9.0.1", "vscode-uri": "^3.0.7", "which": "^3.0.1", - "zod": "^3.21.4" + "zod": "^3.21.4", + "zod-fixture": "^2.5.2" }, "devDependencies": { "@esbuild-plugins/node-globals-polyfill": "^0.2.3", diff --git a/src/api/terraformCloud/account.ts b/src/api/terraformCloud/account.ts index 9ef4dc25f..a626e03a9 100644 --- a/src/api/terraformCloud/account.ts +++ b/src/api/terraformCloud/account.ts @@ -7,7 +7,7 @@ import { makeApi } from '@zodios/core'; import { z } from 'zod'; import { errors } from './errors'; -const accountDetails = z.object({ +export const accountDetails = z.object({ data: z.object({ id: z.string(), type: z.string(), @@ -18,6 +18,8 @@ const accountDetails = z.object({ }), }); +export type Account = z.infer; + export const accountEndpoints = makeApi([ { method: 'get', diff --git a/src/api/terraformCloud/organization.ts b/src/api/terraformCloud/organization.ts index 4f8bc85b7..04cf93eb5 100644 --- a/src/api/terraformCloud/organization.ts +++ b/src/api/terraformCloud/organization.ts @@ -9,7 +9,7 @@ import { paginationMeta, paginationParams } from './pagination'; import { searchQueryParams } from './filter'; import { errors } from './errors'; -const organization = z.object({ +export const organization = z.object({ id: z.string(), attributes: z.object({ 'external-id': z.string(), @@ -19,7 +19,7 @@ const organization = z.object({ export type Organization = z.infer; -const organizations = z.object({ +export const organizations = z.object({ data: z.array(organization), meta: z .object({ @@ -41,7 +41,8 @@ const organizationMemebrship = z.object({ }), }), }); -const organizationMemberships = z.object({ + +export const organizationMemberships = z.object({ data: z.array(organizationMemebrship), }); export type OrganizationMembership = z.infer; diff --git a/src/api/terraformCloud/project.ts b/src/api/terraformCloud/project.ts index 160f100f6..86c8f7474 100644 --- a/src/api/terraformCloud/project.ts +++ b/src/api/terraformCloud/project.ts @@ -9,7 +9,7 @@ import { paginationMeta, paginationParams } from './pagination'; import { searchQueryParams } from './filter'; import { errors } from './errors'; -const project = z.object({ +export const project = z.object({ id: z.string(), attributes: z.object({ name: z.string(), @@ -18,7 +18,7 @@ const project = z.object({ export type Project = z.infer; -const projects = z.object({ +export const projects = z.object({ data: z.array(project), meta: z.object({ pagination: paginationMeta, diff --git a/src/api/terraformCloud/workspace.ts b/src/api/terraformCloud/workspace.ts index 6ec93f1ea..f7c8136dc 100644 --- a/src/api/terraformCloud/workspace.ts +++ b/src/api/terraformCloud/workspace.ts @@ -53,7 +53,7 @@ const workspaceRelationships = z.object({ project: relationship, }); -const workspace = z.object({ +export const workspace = z.object({ id: z.string(), attributes: workspaceAttributes, relationships: workspaceRelationships, @@ -66,7 +66,7 @@ const workspace = z.object({ export type Workspace = z.infer; export type WorkspaceAttributes = z.infer; -const workspaces = z.object({ +export const workspaces = z.object({ data: z.array(workspace), meta: z.object({ pagination: paginationMeta, diff --git a/src/commands/generateBugReport.ts b/src/commands/generateBugReport.ts index 9166cbf99..b5d11d517 100644 --- a/src/commands/generateBugReport.ts +++ b/src/commands/generateBugReport.ts @@ -154,11 +154,9 @@ Outdated:\t${info.outdated} const extensions = vscode.extensions.all .filter((element) => element.packageJSON.isBuiltin === false) .sort((leftside, rightside): number => { - // eslint-disable-next-line @typescript-eslint/no-unsafe-call if (leftside.packageJSON.name.toLowerCase() < rightside.packageJSON.name.toLowerCase()) { return -1; } - // eslint-disable-next-line @typescript-eslint/no-unsafe-call if (leftside.packageJSON.name.toLowerCase() > rightside.packageJSON.name.toLowerCase()) { return 1; } diff --git a/src/extension.ts b/src/extension.ts index 6017a5565..e1565790f 100644 --- a/src/extension.ts +++ b/src/extension.ts @@ -35,6 +35,7 @@ import { TerraformLSCommands } from './commands/terraformls'; import { TerraformCommands } from './commands/terraform'; import * as lsStatus from './status/language'; import { TerraformCloudFeature } from './features/terraformCloud'; +import { setupMockServer, stopMockServer } from './test/e2e/specs/mocks/server'; const id = 'terraform'; const brand = `HashiCorp Terraform`; @@ -55,6 +56,11 @@ let initializationError: ResponseError | Error | undefined = un let crashCount = 0; export async function activate(context: vscode.ExtensionContext): Promise { + // if (config('terraform').get('trace.server') === 'verbose') { + // setupMockServer(); + // } + setupMockServer(); + const manifest = context.extension.packageJSON; reporter = new TelemetryReporter(context.extension.id, manifest.version, manifest.appInsightsKey); context.subscriptions.push(reporter); @@ -240,4 +246,9 @@ export async function deactivate(): Promise { lsStatus.setLanguageServerState(error, false, vscode.LanguageStatusSeverity.Error); } } + + // if (config('terraform').get('trace.server') === 'verbose') { + // stopMockServer(); + // } + stopMockServer(); } diff --git a/src/features/terraformCloud.ts b/src/features/terraformCloud.ts index 521847584..74a29ae7e 100644 --- a/src/features/terraformCloud.ts +++ b/src/features/terraformCloud.ts @@ -19,6 +19,7 @@ import { APIQuickPick } from '../providers/tfc/uiHelpers'; import { TerraformCloudWebUrl } from '../api/terraformCloud'; import { PlanLogContentProvider } from '../providers/tfc/contentProvider'; import { ApplyTreeDataProvider } from '../providers/tfc/applyProvider'; +import { debugChannel } from '../test/e2e/specs/mocks/server'; export class TerraformCloudFeature implements vscode.Disposable { private statusBar: OrganizationStatusBar; @@ -40,6 +41,7 @@ export class TerraformCloudFeature implements vscode.Disposable { this.statusBar = new OrganizationStatusBar(context); authProvider.onDidChangeSessions(async (event) => { + debugChannel.appendLine('Auth provider onDidChangeSessions'); if (event.added && event.added.length > 0) { await vscode.commands.executeCommand('terraform.cloud.organization.picker'); await this.statusBar.show(); @@ -163,6 +165,8 @@ export class TerraformCloudFeature implements vscode.Disposable { vscode.commands.registerCommand('terraform.cloud.organization.picker', async () => { this.reporter.sendTelemetryEvent('tfc-pick-organization'); + debugChannel.appendLine('Picking organization'); + const organizationAPIResource = new OrganizationAPIResource(outputChannel, reporter); const organizationQuickPick = new APIQuickPick(organizationAPIResource); let choice: vscode.QuickPickItem | undefined; @@ -171,6 +175,7 @@ export class TerraformCloudFeature implements vscode.Disposable { while (true) { choice = await organizationQuickPick.pick(false); + debugChannel.appendLine(`Choice: ${choice?.label}`); if (choice === undefined) { // user exited without answering, so don't do anything return; @@ -189,11 +194,15 @@ export class TerraformCloudFeature implements vscode.Disposable { break; } + debugChannel.appendLine('Choice is not undefined, hide organization picker'); // user chose an organization so update the statusbar and make sure its visible organizationQuickPick.hide(); + debugChannel.appendLine('Show status bar with org'); await this.statusBar.show(choice.label); + debugChannel.appendLine('Update workspace view title'); workspaceView.title = `Workspace - (${choice.label})`; + debugChannel.appendLine('Reset project filters'); // project filter should be cleared on org change await vscode.commands.executeCommand('terraform.cloud.workspaces.resetProjectFilter'); // filter reset will refresh workspaces diff --git a/src/providers/tfc/applyProvider.ts b/src/providers/tfc/applyProvider.ts index 8d5a7019d..5311ee4e8 100644 --- a/src/providers/tfc/applyProvider.ts +++ b/src/providers/tfc/applyProvider.ts @@ -9,7 +9,6 @@ import { Writable } from 'stream'; import axios from 'axios'; import TelemetryReporter from '@vscode/extension-telemetry'; -import { TerraformCloudAuthenticationProvider } from './authenticationProvider'; import { ZodiosError } from '@zodios/core'; import { handleAuthError, handleZodiosError } from './uiHelpers'; import { GetChangeActionIcon } from './helpers'; @@ -79,13 +78,13 @@ export class ApplyTreeDataProvider implements vscode.TreeDataProvider { - const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { - createIfNone: false, - }); + // const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { + // createIfNone: false, + // }); - if (session === undefined) { - return; - } + // if (session === undefined) { + // return; + // } try { const result = await axios.get(apply.logReadUrl, { diff --git a/src/providers/tfc/authenticationProvider.ts b/src/providers/tfc/authenticationProvider.ts index a17b24e27..7b9bb85ad 100644 --- a/src/providers/tfc/authenticationProvider.ts +++ b/src/providers/tfc/authenticationProvider.ts @@ -146,6 +146,14 @@ export class TerraformCloudAuthenticationProvider implements vscode.Authenticati }); vscode.window.showInformationMessage(`Hello ${session.account.label}`); }), + vscode.commands.registerCommand('terraform.cloud.logout', async () => { + const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { + createIfNone: false, + }); + if (session) { + await this.removeSession(session.id); + } + }), ); this.sessionPromise = this.sessionHandler.get(); @@ -297,7 +305,7 @@ export class TerraformCloudAuthenticationProvider implements vscode.Authenticati return; } - this._onDidChangeSessions.fire({ added: added, removed: removed, changed: changed }); + // this._onDidChangeSessions.fire({ added: added, removed: removed, changed: changed }); } private async promptForTFCHostname(): Promise { diff --git a/src/providers/tfc/planProvider.ts b/src/providers/tfc/planProvider.ts index 506906bf3..5734d76cb 100644 --- a/src/providers/tfc/planProvider.ts +++ b/src/providers/tfc/planProvider.ts @@ -9,7 +9,7 @@ import { Writable } from 'stream'; import axios from 'axios'; import TelemetryReporter from '@vscode/extension-telemetry'; -import { TerraformCloudAuthenticationProvider } from './authenticationProvider'; +// import { TerraformCloudAuthenticationProvider } from './authenticationProvider'; import { ZodiosError } from '@zodios/core'; import { handleAuthError, handleZodiosError } from './uiHelpers'; import { GetChangeActionIcon, GetDriftChangeActionMessage } from './helpers'; @@ -82,13 +82,13 @@ export class PlanTreeDataProvider implements vscode.TreeDataProvider { - const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { - createIfNone: false, - }); + // const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { + // createIfNone: false, + // }); - if (session === undefined) { - return; - } + // if (session === undefined) { + // return; + // } try { const result = await axios.get(plan.logReadUrl, { diff --git a/src/providers/tfc/runProvider.ts b/src/providers/tfc/runProvider.ts index 9ab3148e7..7572fb2b5 100644 --- a/src/providers/tfc/runProvider.ts +++ b/src/providers/tfc/runProvider.ts @@ -9,7 +9,6 @@ import TelemetryReporter from '@vscode/extension-telemetry'; import semver from 'semver'; import { TerraformCloudWebUrl, apiClient } from '../../api/terraformCloud'; -import { TerraformCloudAuthenticationProvider } from './authenticationProvider'; import { RUN_SOURCE, RunAttributes, TRIGGER_REASON } from '../../api/terraformCloud/run'; import { WorkspaceTreeItem } from './workspaceProvider'; import { GetPlanApplyStatusIcon, GetRunStatusIcon, GetRunStatusMessage, RelativeTimeFormat } from './helpers'; @@ -21,6 +20,7 @@ import { ApplyAttributes } from '../../api/terraformCloud/apply'; import { CONFIGURATION_SOURCE } from '../../api/terraformCloud/configurationVersion'; import { PlanTreeDataProvider } from './planProvider'; import { ApplyTreeDataProvider } from './applyProvider'; +import { appendLine } from '../../test/e2e/specs/mocks/server'; export class RunTreeDataProvider implements vscode.TreeDataProvider, vscode.Disposable { private readonly didChangeTreeData = new vscode.EventEmitter(); @@ -109,13 +109,13 @@ export class RunTreeDataProvider implements vscode.TreeDataProvider, vscode.Disposable { private readonly didChangeTreeData = new vscode.EventEmitter(); @@ -83,13 +83,13 @@ export class WorkspaceTreeDataProvider implements vscode.TreeDataProvider { - const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { - createIfNone: false, - }); + // const session = await vscode.authentication.getSession(TerraformCloudAuthenticationProvider.providerID, [], { + // createIfNone: false, + // }); - if (session === undefined) { - return; - } + // if (session === undefined) { + // return; + // } const organization = this.ctx.workspaceState.get('terraform.cloud.organization', ''); const projectAPIResource = new ProjectsAPIResource(organization, this.outputChannel, this.reporter); @@ -142,13 +142,13 @@ export class WorkspaceTreeDataProvider implements vscode.TreeDataProvider { - it('should be able to load VSCode', async () => { - const workbench = await browser.getWorkbench(); - expect(await workbench.getTitleBar().getTitle()).toContain('[Extension Development Host]'); - }); + let terraformExtension: ExtensionsViewItem; + let activityBar: ActivityBar; + let bottomBarPanel: BottomBarPanel; + + before(async function () { + this.timeout(15000); + // open the extensions view + const view = await (await new ActivityBar().getViewControl('Extensions'))?.openView(); + await view?.getDriver().wait(async function () { + return (await view.getContent().getSections()).length > 0; + }); - it('should load and install our VSCode Extension', async () => { - const extensions = await browser.executeWorkbench((vscodeApi) => { - return vscodeApi.extensions.all; + // we want to find the terraform extension (this project) + // first we need a view section, best place to get started is the 'Installed' section + const extensions = (await view?.getContent().getSection('Installed')) as ExtensionsViewSection; + + // search for the extension, you can use any syntax vscode supports for the search field + // it is best to prepend @installed to the extension name if you don't want to see the results from marketplace + // also, getting the name directly from package.json seem like a good idea + await extensions.getDriver().wait(async function () { + terraformExtension = (await extensions.findItem(`@installed HashiCorp Terraform`)) as ExtensionsViewItem; + return terraformExtension !== undefined; }); - expect(extensions.some((extension) => extension.id === 'hashicorp.terraform')).toBe(true); + + activityBar = new ActivityBar(); + bottomBarPanel = new BottomBarPanel(); }); - it('should show all activity bar items', async () => { - const workbench = await browser.getWorkbench(); - const viewControls = await workbench.getActivityBar().getViewControls(); - expect(await Promise.all(viewControls.map((vc) => vc.getTitle()))).toEqual([ - 'Explorer', - 'Search', - 'Source Control', - 'Run and Debug', - 'Extensions', - 'HashiCorp Terraform', - 'HCP Terraform', - ]); + it('Check the extension info', async () => { + // now we have the extension item, we can check it shows all the fields we want + const author = await terraformExtension.getAuthor(); + const version = await terraformExtension.getVersion(); + + // in this case we are comparing the results against the values in package.json + expect(author).equals('hashicorp'); + expect(version).equals('2.34.2024101517'); }); - // this does not appear to work in CI - // it('should start the ls', async () => { - // const workbench = await browser.getWorkbench(); - // await workbench.executeCommand('workbench.panel.output.focus'); + it('should be able to load VSCode', async () => { + const titleBar = new TitleBar(); + const title = await titleBar.getTitle(); + expect(title).matches(/[Extension Development Host]/); + }); + + it('should show extension activity bar items', async () => { + const controls = await activityBar.getViewControls(); + expect(controls).not.empty; + + // get titles from the controls + const titles = await Promise.all( + controls.map(async (control) => { + return control.getTitle(); + }), + ); - // const bottomBar = workbench.getBottomBar(); - // await bottomBar.maximize(); + // assert a view control named 'Explorer' is present + // the keyboard shortcut is part of the title, so we do a little transformation + expect(titles.some((title) => title.startsWith('HCP Terraform'))).is.true; + expect(titles.some((title) => title.startsWith('HashiCorp Terraform'))).is.true; + }); - // const outputView = await bottomBar.openOutputView(); + // it('should start the ls', async () => { + // const outputView = await bottomBarPanel.openOutputView(); // await outputView.wait(); // await outputView.selectChannel('HashiCorp Terraform'); - // const output = await outputView.getText(); + // await outputView.wait(); + + // const text = await outputView.getText(); - // expect(output.some((element) => element.toLowerCase().includes('dispatching next job'.toLowerCase()))).toBeTruthy(); + // expect(text).to.include('Dispatching next job'); // }); }); diff --git a/src/test/e2e/specs/language/settings.e2e.ts b/src/test/e2e/specs/language/settings.e2e.ts new file mode 100644 index 000000000..cdd568392 --- /dev/null +++ b/src/test/e2e/specs/language/settings.e2e.ts @@ -0,0 +1,35 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import { expect } from 'chai'; +import { ArraySetting, CheckboxSetting, SettingsEditor, Workbench } from 'vscode-extension-tester'; + +describe('Settings Editor', () => { + let settings: SettingsEditor; + + before(async () => { + settings = await new Workbench().openSettings(); + }); + + it('terraform.languageServer.enable should be true by default', async () => { + const setting = await settings.findSetting('Enable', 'Terraform', 'Language Server'); + + const simpleDialogSetting = setting as CheckboxSetting; + expect(await simpleDialogSetting.getValue()).is.true; + + const desc = await simpleDialogSetting.getDescription(); + expect(desc).contains('Enable Terraform Language Server'); + }); + + it('terraform.languageServer.args should have serve by default', async () => { + const argsSetting = (await settings.findSetting('Args', 'Terraform', 'Language Server')) as ArraySetting; + + const args = await argsSetting.getValues(); + expect(args).is.not.empty; + expect(args).to.include('serve'); + + expect(await argsSetting.getDescription()).contains('Arguments to pass to language server binary'); + }); +}); diff --git a/src/test/e2e/specs/language/terraform.e2e..ts b/src/test/e2e/specs/language/terraform.e2e..ts deleted file mode 100644 index 9d1e85cb3..000000000 --- a/src/test/e2e/specs/language/terraform.e2e..ts +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) HashiCorp, Inc. - * SPDX-License-Identifier: MPL-2.0 - */ - -import { StatusBar } from 'wdio-vscode-service'; -import { browser, expect } from '@wdio/globals'; - -import path from 'node:path'; -import { fileURLToPath } from 'url'; - -const __filename = fileURLToPath(import.meta.url); -const __dirname = path.dirname(__filename); - -describe('Terraform language tests', () => { - let statusBar: StatusBar; - - before(async () => { - const workbench = await browser.getWorkbench(); - statusBar = workbench.getStatusBar(); - - const testFile = path.join(__dirname, '../../../', 'fixtures', `sample.tf`); - browser.executeWorkbench((vscode, fileToOpen) => { - vscode.commands.executeCommand('vscode.open', vscode.Uri.file(fileToOpen)); - }, testFile); - }); - - after(async () => { - // TODO: Close the file - }); - - it('can detect correct language', async () => { - expect(await statusBar.getCurrentLanguage()).toContain('Terraform'); - }); - - // it('can detect terraform version', async () => { - // let item: WebdriverIO.Element | undefined; - // await browser.waitUntil( - // async () => { - // const i = await statusBar.getItems(); - // // console.log(i); - - // item = await statusBar.getItem( - // 'Editor Language Status: 0.32.7, Terraform LS, next: 1.6.6, Terraform Installed, next: any, Terraform Required', - // ); - // }, - // { timeout: 10000, timeoutMsg: 'Did not find a version' }, - // ); - - // expect(item).toBeDefined(); - // }); -}); diff --git a/src/test/e2e/specs/language/terraform.e2e.ts b/src/test/e2e/specs/language/terraform.e2e.ts new file mode 100644 index 000000000..5c8d565f2 --- /dev/null +++ b/src/test/e2e/specs/language/terraform.e2e.ts @@ -0,0 +1,52 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import { StatusBar, VSBrowser } from 'vscode-extension-tester'; +import { expect } from 'chai'; + +import path from 'node:path'; + +describe('Terraform language tests', () => { + let statusBar: StatusBar; + + before(async function () { + this.timeout(15_000); + statusBar = new StatusBar(); + // most basic functions of status bar are only available when a file is opened + await VSBrowser.instance.openResources(path.join('src', 'test', 'fixtures', 'sample.tf')); + }); + + it('can detect correct language', async () => { + // retrieve an item from the status bar by label (the text visible on the bar) + // we are looking at a tf file, so we can get the language selection item like so + const item = await statusBar.getItem('Terraform'); + expect(item).not.undefined; + + const language = await statusBar.getCurrentLanguage(); + expect(language).not.undefined; + expect(language).contains('Terraform'); + }); + + // it('can detect terraform version', async () => { + // or get all the available items + // const items = await statusBar.getItems(); + // expect(items.length).greaterThan(2); + // for (const item of items) { + // // console.log(await item.getText()); + // } + // let item: WebdriverIO.Element | undefined; + // await browser.waitUntil( + // async () => { + // const i = await statusBar.getItems(); + // // console.log(i); + // item = await statusBar.getItem( + // 'Editor Language Status: 0.32.7, Terraform LS, next: 1.6.6, Terraform Installed, next: any, Terraform Required', + // ); + // }, + // { timeout: 10000, timeoutMsg: 'Did not find a version' }, + // ); + // expect(item).toBeDefined(); + // }); +}); diff --git a/src/test/e2e/specs/mocks/handlers.ts b/src/test/e2e/specs/mocks/handlers.ts new file mode 100644 index 000000000..047c05d54 --- /dev/null +++ b/src/test/e2e/specs/mocks/handlers.ts @@ -0,0 +1,106 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import * as vscode from 'vscode'; +import { ZodNumber, ZodObject, z } from 'zod'; +import { Fixture, Generator } from 'zod-fixture'; +import { ZodiosPathsByMethod, ZodiosResponseByPath } from '@zodios/core/lib/zodios.types'; +import { ResponseResolver, http, HttpResponse } from 'msw'; +import { setupServer, SetupServerApi } from 'msw/node'; +import { createFixture } from 'zod-fixture'; +import { accountDetails } from '../../../../api/terraformCloud/account'; +import { organization, organizations, organizationMemberships } from '../../../../api/terraformCloud/organization'; +import { workspace, workspaces } from '../../../../api/terraformCloud/workspace'; +import { projects, project } from '../../../../api/terraformCloud/project'; +import { apiClient, TerraformCloudAPIUrl } from '../../../../api/terraformCloud'; +import { mockGetApiClient } from './server'; + +// id: () => `org-${Math.floor(Math.random() * 1000)}`, +// attributes: { +// 'external-id': () => `org-${Math.floor(Math.random() * 1000)}`, +// name: () => `Org ${Math.floor(Math.random() * 1000)}`, +// } +const customOrganizationGenerator = Generator({ + schema: organization, + output: () => { + const orgnum = `${Math.floor(Math.random() * 1000)}`; + return { + id: `org-${orgnum}`, + attributes: { + 'external-id': `org-${orgnum}`, + name: `Org ${orgnum}`, + }, + }; + }, +}); + +const fixture = new Fixture({ seed: 38 }).extend([customOrganizationGenerator]); +const orgs = fixture.fromSchema(organizations); + +// const organizationFixture = createFixture(organizationSchema, { +// generator: customOrganizationGenerator, +// }); + +export const handlers = [ + mockGetApiClient('/account/details', () => { + return HttpResponse.json(createFixture(accountDetails, { seed: 11 })); + }), + mockGetApiClient('/organizations', () => { + // return HttpResponse.json(createFixture(organizations, { seed: 11 })); + return HttpResponse.json(orgs); + }), + mockGetApiClient('/organization-memberships', () => { + // return HttpResponse.json(createFixture(organizations, { seed: 11 })); + return HttpResponse.json(createFixture(organizationMemberships, { seed: 11 })); + }), + // mockGetApiClient('/organizations', () => { + // return HttpResponse.json({ + // data: [ + // { + // id: 'org-1', + // attributes: { + // 'external-id': 'org-1', + // name: 'Org 1', + // }, + // }, + // { + // id: 'org-2', + // attributes: { + // 'external-id': 'org-2', + // name: 'Org 2', + // }, + // }, + // ], + // }); + // }), + + mockGetApiClient('/organizations/:organization_name/workspaces', () => { + return HttpResponse.json(createFixture(workspaces, { seed: 11 })); + }), + mockGetApiClient('/workspaces/:workspace_id', () => { + return HttpResponse.json( + createFixture( + z.object({ + data: workspace, + }), + { seed: 11 }, + ), + ); + }), + + mockGetApiClient('/organizations/:organization_name/projects', () => { + return HttpResponse.json(createFixture(projects, { seed: 11 })); + }), + mockGetApiClient('/projects/:project_id', () => { + return HttpResponse.json( + createFixture( + z.object({ + data: project, + }), + { seed: 11 }, + ), + ); + }), +]; diff --git a/src/test/e2e/specs/mocks/hardcoded.ts b/src/test/e2e/specs/mocks/hardcoded.ts new file mode 100644 index 000000000..90ffbf9c1 --- /dev/null +++ b/src/test/e2e/specs/mocks/hardcoded.ts @@ -0,0 +1,311 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import * as vscode from 'vscode'; +import { ZodNumber, ZodObject, z } from 'zod'; +import { Fixture, Generator } from 'zod-fixture'; +import { ZodiosPathsByMethod, ZodiosResponseByPath } from '@zodios/core/lib/zodios.types'; +import { ResponseResolver, http, HttpResponse } from 'msw'; +import { setupServer, SetupServerApi } from 'msw/node'; +import { createFixture } from 'zod-fixture'; +import { accountDetails } from '../../../../api/terraformCloud/account'; +import { organization, organizations, organizationMemberships } from '../../../../api/terraformCloud/organization'; +import { workspace, workspaces } from '../../../../api/terraformCloud/workspace'; +import { projects, project } from '../../../../api/terraformCloud/project'; +import { apiClient, TerraformCloudAPIUrl } from '../../../../api/terraformCloud'; +import { mockGetApiClient } from './server'; + +export const handlers = [ + mockGetApiClient('/account/details', () => { + return HttpResponse.json({ + data: { + id: 'account-1', + type: 'account', + attributes: { + username: 'user1', + email: 'user1@terraform.org', + }, + }, + }); + }), + mockGetApiClient('/organizations', () => { + return HttpResponse.json({ + data: [ + { + id: 'org-1', + attributes: { + 'external-id': 'org-1', + name: 'Org 1', + }, + }, + { + id: 'org-2', + attributes: { + 'external-id': 'org-2', + name: 'Org 2', + }, + }, + ], + }); + }), + mockGetApiClient('/organization-memberships', () => { + return HttpResponse.json({ + data: [ + { + id: 'ou-VgJgfbDVN3APUm2F', + attributes: { + status: 'active', + }, + relationships: { + organization: { + data: { + id: 'org-1', + }, + }, + }, + }, + { + id: 'ou-VgJgfbDVN3APUm2Fdf', + attributes: { + status: 'active', + }, + relationships: { + organization: { + data: { + id: 'org-2', + }, + }, + }, + }, + ], + }); + }), + + mockGetApiClient('/organizations/:organization_name/projects', () => { + return HttpResponse.json({ + data: [ + { + id: 'proj-1', + attributes: { + name: 'Project 1', + }, + }, + ], + meta: { + pagination: { + 'current-page': 1, + 'page-size': 20, + 'next-page': null, + 'prev-page': null, + 'total-count': 20, + 'total-pages': 0, + }, + }, + }); + }), + mockGetApiClient('/projects/:project_id', () => { + return HttpResponse.json({ + data: { + id: 'proj-1', + attributes: { + name: 'Project 1', + }, + }, + }); + }), + + mockGetApiClient('/organizations/:organization_name/workspaces', () => { + return HttpResponse.json({ + meta: { + pagination: { + 'current-page': 1, + 'page-size': 20, + 'next-page': null, + 'prev-page': null, + 'total-count': 20, + 'total-pages': 0, + }, + }, + included: [], + data: [ + { + id: 'ws-1', + attributes: { + name: 'Workspace 1', + description: 'Workspace 1 description', + environment: 'production', + 'execution-mode': 'agent', + source: 'source', + 'updated-at': new Date('2021-07-01T00:00:00Z'), + 'run-failures': 0, + 'resource-count': 0, + 'terraform-version': '0.12.0', + locked: false, + 'vcs-repo-identifier': 'vcs-repo-identifier', + 'vcs-repo': { + 'repository-http-url': 'https://foo.com/bar', + }, + 'auto-apply': false, + }, + relationships: { + project: { + data: { + id: 'proj-1', + type: 'projects', + }, + }, + 'latest-run': { + data: { + id: 'run-1', + type: 'runs', + }, + }, + }, + links: { + self: '/workspaces/ws-1', + 'self-html': '/workspaces/ws-1', + }, + }, + ], + }); + }), + mockGetApiClient('/workspaces/:workspace_id', () => { + return HttpResponse.json({ + data: { + id: 'ws-1', + attributes: { + name: 'Workspace 1', + description: 'Workspace 1 description', + environment: 'production', + 'execution-mode': 'agent', + source: 'source', + 'updated-at': new Date('2021-07-01T00:00:00Z'), + 'run-failures': 0, + 'resource-count': 0, + 'terraform-version': '0.12.0', + locked: false, + 'vcs-repo-identifier': 'vcs-repo-identifier', + 'vcs-repo': { + 'repository-http-url': 'https://foo.com/bar', + }, + 'auto-apply': false, + }, + relationships: { + project: { + data: { + id: 'proj-1', + type: 'projects', + }, + }, + 'latest-run': { + data: { + id: 'run-1', + type: 'runs', + }, + }, + }, + links: { + self: '/workspaces/ws-1', + 'self-html': '/workspaces/ws-1', + }, + }, + }); + }), + + mockGetApiClient('/workspaces/:workspace_id/runs', () => { + return HttpResponse.json({ + data: [ + { + id: 'run-1', + attributes: { + 'created-at': new Date(Date.UTC(2021, 6, 1, 0, 0, 0)), + message: 'Run 1', + source: 'terraform+cloud', + status: 'planned', + 'trigger-reason': 'manual', + 'terraform-version': '0.12.0', + }, + relationships: { + workspace: { + data: { + id: 'ws-1', + type: 'workspaces', + }, + }, + 'configuration-version': { + data: { + id: 'cv-1', + type: 'configuration-versions', + }, + }, + 'created-by': { + data: { + id: 'user-1', + type: 'users', + }, + }, + }, + }, + ], + meta: { + pagination: { + 'current-page': 1, + 'page-size': 20, + 'next-page': null, + 'prev-page': null, + 'total-count': 20, + 'total-pages': 0, + }, + }, + }); + }), + mockGetApiClient('/runs/:run_id', () => { + return HttpResponse.json({ + data: { + id: 'run-1', + attributes: { + 'created-at': new Date('2021-07-01T00:00:00Z'), + message: 'Run 1', + source: 'terraform+cloud', + status: 'planned', + 'trigger-reason': 'manual', + 'terraform-version': '0.12.0', + }, + relationships: { + plan: { + data: { + id: 'plan-1', + type: 'plans', + }, + }, + apply: { + data: { + id: 'apply-1', + type: 'applies', + }, + }, + workspace: { + data: { + id: 'ws-1', + type: 'workspaces', + }, + }, + 'configuration-version': { + data: { + id: 'cv-1', + type: 'configuration-versions', + }, + }, + 'created-by': { + data: { + id: 'user-1', + type: 'users', + }, + }, + }, + }, + }); + }), +]; diff --git a/src/test/e2e/specs/mocks/server.ts b/src/test/e2e/specs/mocks/server.ts new file mode 100644 index 000000000..d18999fda --- /dev/null +++ b/src/test/e2e/specs/mocks/server.ts @@ -0,0 +1,51 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import * as vscode from 'vscode'; +import { ZodiosPathsByMethod, ZodiosResponseByPath } from '@zodios/core/lib/zodios.types'; +import { ResponseResolver, http } from 'msw'; +import { setupServer, SetupServerApi } from 'msw/node'; +import { apiClient, TerraformCloudAPIUrl } from '../../../../api/terraformCloud'; +import { handlers } from './hardcoded'; + +type Api = typeof apiClient.api; + +export function mockGetApiClient>( + path: Path, + // eslint-disable-next-line @typescript-eslint/ban-types + resolver: ResponseResolver<{}, never, Awaited>>, +) { + return http.get(`${TerraformCloudAPIUrl}${path}`, resolver); +} + +let server: SetupServerApi; +export let debugChannel: vscode.OutputChannel; + +export function setupMockServer() { + debugChannel = vscode.window.createOutputChannel('MSW Debug Channel'); + + server = setupServer(...handlers); + + server.events.on('request:unhandled', ({ request }) => { + debugChannel.appendLine( + `Intercepted a request without a matching request handler: ${request.method} ${request.url}`, + ); + }); + + server.events.on('response:mocked', ({ request, response }) => { + debugChannel.appendLine( + `Outgoing request "${request.method} ${request.url}" received mock response: ${response.status} ${response.statusText}`, + ); + }); + server.listen(); +} + +export function stopMockServer() { + server.close(); +} + +export function appendLine(line: string) { + debugChannel?.appendLine(line); +} diff --git a/src/test/e2e/specs/views/hcp.e2e.ts b/src/test/e2e/specs/views/hcp.e2e.ts new file mode 100644 index 000000000..9472bf079 --- /dev/null +++ b/src/test/e2e/specs/views/hcp.e2e.ts @@ -0,0 +1,127 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import { + ActivityBar, + BottomBarPanel, + DefaultTreeSection, + EditorView, + InputBox, + ModalDialog, + SideBarView, + ViewContent, + ViewTitlePart, + VSBrowser, + Workbench, +} from 'vscode-extension-tester'; +import * as path from 'path'; +import { expect } from 'chai'; + +describe('HCP tree view tests', () => { + let titlePart: ViewTitlePart; + let content: ViewContent; + let workbench: Workbench; + let workspaceTree: DefaultTreeSection; + let runTree: DefaultTreeSection; + + before(async function () { + await VSBrowser.instance.openResources(path.join('src', 'test', 'fixtures')); + + (await new ActivityBar().getViewControl('HCP Terraform'))?.openView(); + + const view = new SideBarView(); + titlePart = view.getTitlePart(); + content = view.getContent(); + + workbench = new Workbench(); + }); + + it('should have correct title', async () => { + const title = await titlePart.getTitle(); + expect(title.toLowerCase()).equals('hcp terraform'); + }); + + it('should login', async () => { + await workbench.executeCommand('HCP Terraform: Login'); + + const dialog = new ModalDialog(); + const buttons = await dialog.getButtons(); + expect(buttons.length).equals(2); + await dialog.pushButton('Allow'); + + const input = await InputBox.create(); + + // simulate connecting to HCP app.terraform.io + await input.setText('Default HCP Terraform instance'); + await input.confirm(); + + // use the existing token workflow instead of file + await input.setText('Existing user token'); + await input.confirm(); + + // we have mocked the api so the token doesn't matter + await input.setText('fdsfdsdssfgdgdgdfgdsfagdfagdfrergebvbvtrhge'); + await input.confirm(); + + await input.selectQuickPick('Org 1'); + }); + + it('should have workspaces', async () => { + // title should change to Org selected + workspaceTree = (await content.getSection('Workspace - (Org 1)')) as DefaultTreeSection; + expect(workspaceTree).is.not.undefined; + expect(await workspaceTree?.isDisplayed()).is.true; + + const items = await workspaceTree.getVisibleItems(); + + const labels = await Promise.all(items.map((item) => item.getLabel())); + expect(labels).contains('Workspace 1'); + + const item = await workspaceTree.findItem('Workspace 1'); + expect(item).not.undefined; + + expect(await item?.getLabel()).equals('Workspace 1'); + expect(await item?.getDescription()).equals('[Project 1]'); + expect(await item?.getTooltip()).equals('Workspace 1 [Project 1]'); + }); + + it('should show a run when a workspace is clicked', async () => { + const item = await workspaceTree.findItem('Workspace 1'); + expect(item).not.undefined; + await item?.select(); + await item?.click(); + + runTree = (await content.getSection('Runs')) as DefaultTreeSection; + expect(runTree).is.not.undefined; + expect(await runTree?.isDisplayed()).is.true; + + // wait for the run to load otherwise the test will fail + await VSBrowser.instance.driver.sleep(100); + + const runItem = await runTree.findItem('Run 1'); + expect(runItem).not.undefined; + await runItem?.select(); + await runItem?.click(); + expect(await runItem?.getLabel()).equals('Run 1'); + expect(await runItem?.getDescription()).contains('manual'); + expect(await runItem?.getTooltip()).contains('Run 1 manual'); + }); + + async function logOutputChannels() { + // wait a bit for the output channels to be populated + await VSBrowser.instance.driver.sleep(100); + + const bottomBar = new BottomBarPanel(); + await bottomBar.toggle(true); + const view = await bottomBar.openOutputView(); + const channels = ['MSW Debug Channel', 'HCP Terraform', 'HashiCorp Authentication', 'Extension Host']; + for (const channel of channels) { + await view.selectChannel(channel); + console.log(`-------------${channel}---------------------`); + console.log(await view.getText()); + console.log(`-------------${channel}---------------------`); + } + } +}); diff --git a/src/test/e2e/specs/views/terraform.e2e.ts b/src/test/e2e/specs/views/terraform.e2e.ts index 408dae866..366ba5218 100644 --- a/src/test/e2e/specs/views/terraform.e2e.ts +++ b/src/test/e2e/specs/views/terraform.e2e.ts @@ -2,124 +2,93 @@ * Copyright (c) HashiCorp, Inc. * SPDX-License-Identifier: MPL-2.0 */ -import { browser, expect } from '@wdio/globals'; -import { Workbench, CustomTreeItem, SideBarView, ViewSection, ViewControl } from 'wdio-vscode-service'; -import path from 'node:path'; -import { fileURLToPath } from 'url'; +import { + ActivityBar, + BottomBarPanel, + DefaultTreeSection, + EditorView, + InputBox, + ModalDialog, + SideBarView, + ViewContent, + ViewTitlePart, + VSBrowser, + Workbench, +} from 'vscode-extension-tester'; +import * as path from 'path'; +import { expect } from 'chai'; + +describe('Terraform tree view tests', () => { + let titlePart: ViewTitlePart; + let content: ViewContent; + let workbench: Workbench; -const __filename = fileURLToPath(import.meta.url); -const __dirname = path.dirname(__filename); + before(async function () { + // await VSBrowser.instance.openResources(path.join('src', 'test', 'fixtures')); -describe('Terraform ViewContainer', function () { - this.retries(3); - let workbench: Workbench; + (await new ActivityBar().getViewControl('Terraform'))?.openView(); - before(async () => { - workbench = await browser.getWorkbench(); - }); + const view = new SideBarView(); + titlePart = view.getTitlePart(); + content = view.getContent(); - after(async () => { - // TODO: Close the file + workbench = new Workbench(); }); - it('should have terraform viewcontainer', async () => { - const viewContainers = await workbench.getActivityBar().getViewControls(); - const titles = await Promise.all(viewContainers.map((vc) => vc.getTitle())); - expect(titles).toContain('HashiCorp Terraform'); + it('should have correct title', async () => { + const title = await titlePart.getTitle(); + expect(title.toLowerCase()).equals('hashicorp terraform'); }); - describe('in an terraform project', () => { + describe('in a terraform project', () => { before(async () => { - const testFile = path.join(__dirname, '../../../', 'fixtures', `sample.tf`); - browser.executeWorkbench((vscode, fileToOpen) => { - vscode.commands.executeCommand('vscode.open', vscode.Uri.file(fileToOpen)); - }, testFile); + await VSBrowser.instance.openResources(path.join('src', 'test', 'fixtures', 'sample.tf')); }); - after(async () => { - // TODO: close the file - }); - - describe('providers view', () => { - let terraformViewContainer: ViewControl | undefined; - let openViewContainer: SideBarView | undefined; - let callSection: ViewSection | undefined; - let items: CustomTreeItem[]; - + describe('proividrs view', () => { + let providersView: DefaultTreeSection; before(async () => { - terraformViewContainer = await workbench.getActivityBar().getViewControl('HashiCorp Terraform'); - await terraformViewContainer?.wait(); - await terraformViewContainer?.openView(); - openViewContainer = workbench.getSideBar(); + providersView = (await content.getSection('Providers')) as DefaultTreeSection; + expect(providersView).is.not.undefined; + expect(await providersView?.isDisplayed()).is.true; }); - it('should have providers view', async () => { - callSection = await openViewContainer?.getContent().getSection('PROVIDERS'); - expect(callSection).toBeDefined(); - }); + it('should have provider calls', async () => { + const items = await providersView.getVisibleItems(); - it('should include all providers', async () => { - callSection = await openViewContainer?.getContent().getSection('PROVIDERS'); - - await browser.waitUntil( - async () => { - const provider = await callSection?.getVisibleItems(); - if (!provider) { - return false; - } - - if (provider.length > 0) { - items = provider as CustomTreeItem[]; - return true; - } - }, - { timeout: 3_000, timeoutMsg: 'Never found any providers' }, - ); - - const labels = await Promise.all(items.map((vi) => vi.getLabel())); - expect(labels).toEqual(['-/vault', 'hashicorp/google']); + const labels = await Promise.all(items.map((item) => item.getLabel())); + expect(labels).contains('hashicorp/google'); + + const item = await providersView.findItem('-/vault'); + expect(item).not.undefined; + + expect(await item?.getLabel()).equals('-/vault'); + // expect(await item?.getDescription()).equals('[Project 1]'); + expect(await item?.getTooltip()).equals('registry.terraform.io/-/vault '); }); }); describe('calls view', () => { - let terraformViewContainer: ViewControl | undefined; - let openViewContainer: SideBarView | undefined; - let callSection: ViewSection | undefined; - let items: CustomTreeItem[]; - + let callsView: DefaultTreeSection; before(async () => { - terraformViewContainer = await workbench.getActivityBar().getViewControl('HashiCorp Terraform'); - await terraformViewContainer?.wait(); - await terraformViewContainer?.openView(); - openViewContainer = workbench.getSideBar(); + callsView = (await content.getSection('Module Calls')) as DefaultTreeSection; + expect(callsView).is.not.undefined; + expect(await callsView?.isDisplayed()).is.true; }); - it('should have module calls view', async () => { - callSection = await openViewContainer?.getContent().getSection('MODULE CALLS'); - expect(callSection).toBeDefined(); - }); + it('should have module calls', async () => { + const items = await callsView.getVisibleItems(); + + const labels = await Promise.all(items.map((item) => item.getLabel())); + expect(labels).contains('compute'); + + const item = await callsView.findItem('local'); + expect(item).not.undefined; - it('should include all module calls', async () => { - callSection = await openViewContainer?.getContent().getSection('MODULE CALLS'); - - await browser.waitUntil( - async () => { - const calls = await callSection?.getVisibleItems(); - if (!calls) { - return false; - } - - if (calls.length > 0) { - items = calls as CustomTreeItem[]; - return true; - } - }, - { timeout: 3_000, timeoutMsg: 'Never found any modules' }, - ); - - const labels = await Promise.all(items.map((vi) => vi.getLabel())); - expect(labels).toEqual(['compute', 'local']); + expect(await item?.getLabel()).equals('local'); + // expect(await item?.getDescription()).equals('[Project 1]'); + expect(await item?.getTooltip()).equals('./modules'); }); }); }); diff --git a/src/test/e2e/specs/views/tfc.e2e.ts b/src/test/e2e/specs/views/tfc.e2e.ts deleted file mode 100644 index 496bccf05..000000000 --- a/src/test/e2e/specs/views/tfc.e2e.ts +++ /dev/null @@ -1,104 +0,0 @@ -/** - * Copyright (c) HashiCorp, Inc. - * SPDX-License-Identifier: MPL-2.0 - */ -import { browser, expect } from '@wdio/globals'; -import { fail } from 'assert'; -import { Workbench, SideBarView, ViewSection, ViewControl, WelcomeContentButton } from 'wdio-vscode-service'; -import { Key } from 'webdriverio'; - -let workbench: Workbench; -let terraformViewContainer: SideBarView; -let callSection: ViewSection; - -describe('TFC ViewContainer', function () { - this.retries(3); - - beforeEach(async () => { - workbench = await browser.getWorkbench(); - }); - - it('should have TFC viewcontainer', async () => { - const viewContainers = await workbench.getActivityBar().getViewControls(); - const titles = await Promise.all(viewContainers.map((vc) => vc.getTitle())); - expect(titles).toContain('HCP Terraform'); - }); - - describe('not logged in', () => { - let terraformViewControl: ViewControl | undefined; - - beforeEach(async () => { - terraformViewControl = await workbench.getActivityBar().getViewControl('HCP Terraform'); - expect(terraformViewControl).toBeDefined(); - await terraformViewControl?.wait(); - await terraformViewControl?.openView(); - terraformViewContainer = workbench.getSideBar(); - }); - - it('should have workspaces view', async () => { - callSection = await terraformViewContainer.getContent().getSection('WORKSPACES'); - expect(callSection).toBeDefined(); - - const welcome = await callSection.findWelcomeContent(); - - const text = await welcome?.getTextSections(); - expect(text).toContain('In order to use HCP Terraform features, you need to be logged in'); - }); - - it('should have runs view', async () => { - callSection = await terraformViewContainer.getContent().getSection('RUNS'); - expect(callSection).toBeDefined(); - }); - }); - - describe('logged in', () => { - let terraformViewControl: ViewControl | undefined; - - beforeEach(async () => { - terraformViewControl = await workbench.getActivityBar().getViewControl('HCP Terraform'); - expect(terraformViewControl).toBeDefined(); - await terraformViewControl?.wait(); - await terraformViewControl?.openView(); - terraformViewContainer = workbench.getSideBar(); - }); - - it('should login', async () => { - callSection = await terraformViewContainer.getContent().getSection('WORKSPACES'); - expect(callSection).toBeDefined(); - - const welcome = await callSection.findWelcomeContent(); - - const text = await welcome?.getTextSections(); - expect(text).toContain('In order to use HCP Terraform features, you need to be logged in'); - - const buttons = await welcome?.getButtons(); - expect(buttons).toHaveLength(1); - if (!buttons) { - fail('No buttons found'); - } - - let loginButton: WelcomeContentButton | undefined; - for (const button of buttons) { - const buttonText = await button.getTitle(); - if (buttonText.toLowerCase().includes('login')) { - loginButton = button; - } - } - if (!loginButton) { - fail("Couldn't find the login button"); - } - - (await loginButton.elem).click(); - - // detect modal and click Allow - browser.keys([Key.Enter]); - - // detect quickpick and select Existing user token - browser.keys(['ArrowDown', Key.Enter]); - - // TODO: enter token in input box and hit enter - - // TODO: verify you are logged in - }); - }); -}); diff --git a/uitest.mjs b/uitest.mjs new file mode 100644 index 000000000..6bfd5fa3b --- /dev/null +++ b/uitest.mjs @@ -0,0 +1,39 @@ +/** + * Copyright (c) HashiCorp, Inc. + * SPDX-License-Identifier: MPL-2.0 + */ + +import { ExTester, ReleaseQuality } from 'vscode-extension-tester'; + +async function runTests() { + let code_version; + let code_type; + + if (process.env.VSCODE_VERSION === 'insiders') { + code_type = ReleaseQuality.Insider; + } else if (process.env.VSCODE_VERSION === 'stable') { + code_type = ReleaseQuality.Stable; + } else { + code_version = process.env.VSCODE_VERSION; + } + + console.log(`Running tests for ${code_version} ${code_type} version of VS Code`); + + const tester = new ExTester('.test-storage', code_type, '.test-extensions', false); + await tester.setupAndRunTests( + 'out/test/e2e/specs/**/*.e2e.js', + code_version, + { + installDependencies: false, + }, + { + settings: 'src/test/e2e/settings.json', + cleanup: true, + config: 'src/test/e2e/.mocharc.js', + logLevel: 'info', + resources: [], + }, + ); +} + +runTests().catch(console.error);