mirror of
https://github.com/usebruno/bruno.git
synced 2026-06-27 22:54:07 +00:00
398 lines
17 KiB
TypeScript
398 lines
17 KiB
TypeScript
import { test, expect } from '../../../playwright';
|
|
import path from 'path';
|
|
import { Page } from '@playwright/test';
|
|
import { importCollection, createEnvironment, closeAllCollections, addRowToActiveTab, saveEnvironment, deleteAllGlobalEnvironments } from '../../utils/page';
|
|
import { buildCommonLocators } from '../../utils/page/locators';
|
|
|
|
const variablesTab = (page: Page) => buildCommonLocators(page).environment.variablesTab();
|
|
const secretsTab = (page: Page) => buildCommonLocators(page).environment.secretsTab();
|
|
const varRow = (page: Page, name: string) => buildCommonLocators(page).environment.varRow(name);
|
|
const saveTab = (page: Page) => buildCommonLocators(page).environment.saveTab();
|
|
const tabDraftIcon = (page: Page) => page.locator('.request-tab.active').getByTestId('tab-draft-icon');
|
|
|
|
const searchEnv = async (page: Page, query: string) => {
|
|
const input = page.locator('.search-input');
|
|
if ((await input.count()) === 0) {
|
|
await page.locator('.env-search-container button[title="Search"]').click();
|
|
await input.waitFor({ state: 'visible' });
|
|
}
|
|
await input.fill(query);
|
|
};
|
|
|
|
const deleteRow = async (page: Page, name: string) => {
|
|
await varRow(page, name).locator('button:has(.icon-tabler-trash)').click();
|
|
};
|
|
|
|
const collectionFile = path.join(__dirname, '..', 'create-environment', 'fixtures', 'bruno-collection.json');
|
|
|
|
test.describe('Environment Variables / Secrets tab separation', () => {
|
|
test.afterEach(async ({ page }) => {
|
|
await closeAllCollections(page);
|
|
});
|
|
|
|
test('keeps variables and secrets on their own tabs', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('var-secret-tabs'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Tab Separation Env', 'collection');
|
|
|
|
await test.step('Add a variable on the Variables tab', async () => {
|
|
await expect(variablesTab(page)).toHaveClass(/active/);
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
});
|
|
|
|
await test.step('Add a secret on the Secrets tab', async () => {
|
|
await secretsTab(page).click();
|
|
await expect(secretsTab(page)).toHaveClass(/active/);
|
|
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
});
|
|
|
|
await test.step('Variables tab shows only the variable, not the secret', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(variablesTab(page)).toHaveClass(/active/);
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
});
|
|
});
|
|
|
|
test('saves variables and secrets independently and persists both', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('var-secret-save'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Independent Save Env', 'collection');
|
|
|
|
await test.step('Add and save a variable on the Variables tab', async () => {
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await saveTab(page).click();
|
|
// newest toast: the two back-to-back saves can briefly show two identical toasts
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
});
|
|
|
|
await test.step('Add and save a secret on the Secrets tab', async () => {
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
});
|
|
|
|
await test.step('Saving the Secrets tab did not wipe the saved variable', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
|
|
await secretsTab(page).click();
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
});
|
|
});
|
|
|
|
test('common save icon persists both variables and secrets at once', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('var-secret-save-all'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Save All Env', 'collection');
|
|
|
|
await test.step('Add a variable on the Variables tab without saving', async () => {
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
});
|
|
|
|
await test.step('Add a secret on the Secrets tab without saving', async () => {
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
});
|
|
|
|
await test.step('The environment tab shows the unsaved-changes dot while drafts exist', async () => {
|
|
await expect(tabDraftIcon(page)).toBeVisible();
|
|
});
|
|
|
|
await test.step('The common save icon saves both tabs in a single click', async () => {
|
|
await saveEnvironment(page);
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
});
|
|
|
|
await test.step('Both the variable and the secret are persisted on their tabs', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
|
|
await secretsTab(page).click();
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
});
|
|
|
|
await test.step('The unsaved-changes dot is gone once both tabs are saved', async () => {
|
|
await expect(tabDraftIcon(page)).not.toBeVisible();
|
|
});
|
|
|
|
await test.step('A second save reports nothing to save, proving both were committed', async () => {
|
|
await saveEnvironment(page);
|
|
await expect(page.getByText('No changes to save')).toBeVisible();
|
|
});
|
|
});
|
|
|
|
test('search is scoped to the active tab', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('var-secret-search'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Search Scope Env', 'collection');
|
|
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await test.step('Searching the Variables tab never surfaces the secret', async () => {
|
|
await variablesTab(page).click();
|
|
await searchEnv(page, 'host');
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
|
|
await searchEnv(page, 'apiToken');
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
await expect(page.getByText('No results found')).toBeVisible();
|
|
});
|
|
|
|
await test.step('Searching the Secrets tab never surfaces the variable', async () => {
|
|
await secretsTab(page).click();
|
|
await searchEnv(page, 'apiToken');
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
|
|
await searchEnv(page, 'host');
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
await expect(page.getByText('No results found')).toBeVisible();
|
|
});
|
|
|
|
// The search query is stored in Redux and persists across environments, so clear
|
|
// it before the test ends — otherwise it filters the next test's table to "No
|
|
// results" and the empty "Name" row never renders.
|
|
await searchEnv(page, '');
|
|
});
|
|
|
|
test('deleting on one tab leaves the other tab untouched', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('var-secret-delete'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Delete Scope Env', 'collection');
|
|
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await test.step('Delete the secret on the Secrets tab', async () => {
|
|
await deleteRow(page, 'apiToken');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
});
|
|
|
|
await test.step('The variable on the Variables tab is untouched', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
});
|
|
});
|
|
});
|
|
|
|
test.describe('Global Environment Variables / Secrets tab separation', () => {
|
|
test.afterEach(async ({ page }) => {
|
|
await deleteAllGlobalEnvironments(page);
|
|
await closeAllCollections(page);
|
|
});
|
|
|
|
test('keeps variables and secrets on their own tabs', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('global-var-secret-tabs'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Global Tab Separation Env', 'global');
|
|
|
|
await test.step('Add a variable on the Variables tab', async () => {
|
|
await expect(variablesTab(page)).toHaveClass(/active/);
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
});
|
|
|
|
await test.step('Add a secret on the Secrets tab', async () => {
|
|
await secretsTab(page).click();
|
|
await expect(secretsTab(page)).toHaveClass(/active/);
|
|
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
});
|
|
|
|
await test.step('Variables tab shows only the variable, not the secret', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(variablesTab(page)).toHaveClass(/active/);
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
});
|
|
});
|
|
|
|
test('saves variables and secrets independently and persists both', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('global-var-secret-save'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Global Independent Save Env', 'global');
|
|
|
|
await test.step('Add and save a variable on the Variables tab', async () => {
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await saveTab(page).click();
|
|
// newest toast: the two back-to-back saves can briefly show two identical toasts
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
});
|
|
|
|
await test.step('Add and save a secret on the Secrets tab', async () => {
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
});
|
|
|
|
await test.step('Saving the Secrets tab did not wipe the saved variable', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
|
|
await secretsTab(page).click();
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
});
|
|
});
|
|
|
|
test('common save icon persists both variables and secrets at once', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('global-var-secret-save-all'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Global Save All Env', 'global');
|
|
|
|
await test.step('Add a variable on the Variables tab without saving', async () => {
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
});
|
|
|
|
await test.step('Add a secret on the Secrets tab without saving', async () => {
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
});
|
|
|
|
await test.step('The environment tab shows the unsaved-changes dot while drafts exist', async () => {
|
|
await expect(tabDraftIcon(page)).toBeVisible();
|
|
});
|
|
|
|
await test.step('The common save icon saves both tabs in a single click', async () => {
|
|
await saveEnvironment(page);
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
});
|
|
|
|
await test.step('Both the variable and the secret are persisted on their tabs', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
|
|
await secretsTab(page).click();
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
});
|
|
|
|
await test.step('The unsaved-changes dot is gone once both tabs are saved', async () => {
|
|
await expect(tabDraftIcon(page)).not.toBeVisible();
|
|
});
|
|
|
|
await test.step('A second save reports nothing to save, proving both were committed', async () => {
|
|
await saveEnvironment(page);
|
|
await expect(page.getByText('No changes to save')).toBeVisible();
|
|
});
|
|
});
|
|
|
|
test('search is scoped to the active tab', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('global-var-secret-search'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Global Search Scope Env', 'global');
|
|
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await test.step('Searching the Variables tab never surfaces the secret', async () => {
|
|
await variablesTab(page).click();
|
|
await searchEnv(page, 'host');
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
|
|
await searchEnv(page, 'apiToken');
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
await expect(page.getByText('No results found')).toBeVisible();
|
|
});
|
|
|
|
await test.step('Searching the Secrets tab never surfaces the variable', async () => {
|
|
await secretsTab(page).click();
|
|
await searchEnv(page, 'apiToken');
|
|
await expect(varRow(page, 'apiToken')).toBeVisible();
|
|
|
|
await searchEnv(page, 'host');
|
|
await expect(varRow(page, 'host')).toHaveCount(0);
|
|
await expect(page.getByText('No results found')).toBeVisible();
|
|
});
|
|
|
|
// The search query is stored in Redux and persists across environments, so clear
|
|
// it before the test ends — otherwise it filters the next test's table to "No
|
|
// results" and the empty "Name" row never renders.
|
|
await searchEnv(page, '');
|
|
});
|
|
|
|
test('deleting on one tab leaves the other tab untouched', async ({ page, createTmpDir }) => {
|
|
await importCollection(page, collectionFile, await createTmpDir('global-var-secret-delete'), {
|
|
expectedCollectionName: 'test_collection'
|
|
});
|
|
|
|
await createEnvironment(page, 'Global Delete Scope Env', 'global');
|
|
|
|
await addRowToActiveTab(page, 'host', 'https://echo.usebruno.com');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await secretsTab(page).click();
|
|
await addRowToActiveTab(page, 'apiToken', 'super-secret-token-12345');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
|
|
await test.step('Delete the secret on the Secrets tab', async () => {
|
|
await deleteRow(page, 'apiToken');
|
|
await saveTab(page).click();
|
|
await expect(page.getByText('Changes saved successfully').last()).toBeVisible();
|
|
await expect(varRow(page, 'apiToken')).toHaveCount(0);
|
|
});
|
|
|
|
await test.step('The variable on the Variables tab is untouched', async () => {
|
|
await variablesTab(page).click();
|
|
await expect(varRow(page, 'host')).toBeVisible();
|
|
});
|
|
});
|
|
});
|