Skip to content

Commit 87f7426

Browse files
authored
feat(variables): persist scripted variable changes by default + re-enable disabled scripting APIs (usebruno#8315)
* feat(variables): add variable persistence with scripting feat(collections): implement script-driven update for collection variables, ensuring direct root modification and draft synchronization feat(collections): enhance script variable management with baseline tracking and draft preservation * feat(variables): add runtime variable updates and optimize disk writes by implementing dirty flags fix(collections): handle errors during environment persistence in script execution feat(collections): implement baseline clearing for script execution and optimize variable update handling feat(tests): add default persistence tests for environment variables and update runtime variable handling refactor(collections): streamline variable update handling and improve draft management by removing redundant comments and optimizing code clarity test(collection-vars): add verification for draft edits and script variable visibility in collection settings UI refactor(collection-vars): update header value selection logic for improved clarity and accuracy in draft isolation tests * feat(global-environments): enhance global environment updates to resolve stale active UIDs and improve persistence logic - Updated the `updateGlobalEnvironments` reducer to handle stale active UIDs by matching against environment names. - Improved the logic for setting global environments and active UIDs to ensure consistency after disk reloads. - Removed outdated tests related to persisted values in favor of more relevant assertions for environment variable handling. * feat(variables): enhance typed value handling and persistence in global and collection environments - Added tests to infer data types (number, boolean, object) when setting environment and collection variables. - Updated the logic to preserve existing data types when variables are not modified by scripts. - Implemented dirty flags to track changes in typed variables, ensuring accurate persistence across sessions. - Refactored related tests to verify the correct behavior of typed variables in various scenarios. * refactor(variables): streamline data type inference and enhance deletion methods - Removed redundant data type inference logic from global and collection variable updates to simplify the codebase. - Updated deletion methods in the Bru class to use Object.keys for improved resilience against user-defined properties. - Added tests to ensure deletion methods function correctly even when properties are shadowed. - Enhanced clarity in draft merge tests by standardizing keyboard shortcuts for selecting all text. * fix(tests): correct variable naming and improve environment panel interactions - Updated test cases to reflect the correct variable name 'wasSaved' instead of 'was-saved'. - Modified environment panel interaction to remove forced click, enhancing test reliability. - Added a utility function to close the environment panel in safe mode tests for better readability and maintainability. * feat(runtime): enhance variable management and cleanup logic - Introduced a new method to clear script-driven variable baselines for collections, ensuring no stale data leaks into new requests. - Updated the handling of runtime variables in the Bru class to track changes with a new dirty flag, improving state management. - Refactored the application of script environment variables to prevent direct mutations, ensuring immutability and cleaner state updates. - Enhanced the response handling in the script runtime to conditionally include runtime variables based on their dirty state. * feat(variables): improve request handling and state management for collections and environments - Enhanced event listeners to clear global environment baselines on both 'testrun-started' and 'request-queued' events, preventing stale data issues. - Updated global environment and collection variable update events to ignore stale updates from superseded requests, ensuring accurate state management. - Refactored the Bru class to optimize variable management, including checks for existing keys before updates and deletions, improving performance and reliability. - Introduced request UID tracking to maintain consistency across variable updates during concurrent requests. * refactor(collections): update action to clear script variable baselines - Replaced the dispatch of `_clearScriptGlobalEnvBaseline` with `clearScriptVariableBaselines` to improve clarity and maintainability in the Redux action handling for collections. * feat(environments): introduce getScriptModifiedKeys utility for improved variable management - Added a new utility function, `getScriptModifiedKeys`, to identify keys modified by scripts relative to a baseline, enhancing the handling of data types during variable updates. - Updated the application of script environment variables to prevent overwriting user-defined draft changes during no-op writes. - Refactored related logic in collections and global environments to utilize the new utility, ensuring accurate state management and improved clarity in the Redux slices. * refactor(global-environments): simplify active UID resolution logic in updateGlobalEnvironments reducer - Streamlined the logic for resolving the active global environment UID by consolidating conditions into a more concise format. - Removed outdated comments to enhance code clarity and maintainability. - Updated tests to ensure accurate resolution of active UIDs based on incoming environment data. * refactor(tests): remove outdated comments and streamline environment variable row expectations - Eliminated comments related to state sync and inference issues to enhance code clarity. - Adjusted expectations for environment variable row rendering in tests, focusing on relevant assertions. * feat(tests): add comprehensive tests for secret variable persistence in environments - Introduced new test cases to validate the preservation of secret variables when updated via scripts in both collection and global environments. - Implemented tests to ensure that secret values are encrypted before storage and can be correctly decrypted for subsequent requests. - Added fixtures and environment configurations for testing secret variable behavior in both bru and yml formats. - Enhanced utility functions for managing environment configurations and interactions within the test suite. * feat(tests): enhance environment variable tests and add global variable persistence - Updated MultiLineEditor and SingleLineEditor components to include data-testid for secret reveal toggle buttons, improving testability. - Introduced new tests for global environment variable persistence, ensuring non-secret variables survive app restarts and are correctly interpolated. - Added fixtures for workspace and collections to support the new global variable tests, enhancing the overall test coverage for environment management. - Refactored utility functions to streamline interactions with environment variables in tests. * refactor(collections): optimize environment and collection saving logic - Simplified the persistence logic for active environments by directly constructing the environment copy, reducing unnecessary cloning. - Updated the collection saving process to utilize the fresh collection state, ensuring accurate data is saved without drafts. - Enhanced error handling during the save operations to improve reliability and maintainability. * feat(tests): implement collection variable persistence tests - Added multiple test cases to validate the persistence of collection variables across app restarts, including typed values and multiple variable settings. - Created new fixtures for collection variables to support the tests, ensuring accurate simulation of variable management scenarios. - Enhanced the existing collection management logic to ensure that variables are correctly set and deleted as per the test requirements. * feat(tests): add tests for typed global environment variable persistence - Introduced a new test suite to validate the persistence of typed global environment variables across app restarts, ensuring correct data types are maintained. - Created a fixture for the test collection to simulate setting global variables with various data types, including number, boolean, object, and string. - Enhanced the test logic to verify that the environment file reflects the correct state before and after application restarts. * fix(tests): update request tab close interaction in variable persistence tests * fix(tests): improve hover interaction for collection actions in runner tests - Updated the hover logic for revealing collection actions to handle sidebar re-renders more reliably. - Replaced one-shot hover with a polling mechanism to ensure visibility of actions, enhancing test stability. * refactor(environments): streamline environment variable handling and remove ephemeral metadata logic - Simplified the comparison logic for environment variables by removing unnecessary ephemeral metadata handling. - Updated the saving process to directly use the environment variables without stripping metadata, enhancing clarity and maintainability. - Removed outdated comments and unused utility functions related to ephemeral variables, improving code cleanliness. * fix(ipc): update persistActiveEnvironment to handle requestUid for stale updates - Modified the persistActiveEnvironment function to accept a requestUid parameter, allowing for better management of stale updates. - Enhanced the logic to prevent disk writes for superseded requests, improving data integrity during environment persistence. * refactor(bru): remove unused envName variable in deleteAllEnvVars method - Eliminated the envName variable from the deleteAllEnvVars method, simplifying the logic for deleting environment variables. - Cleaned up the method by removing unnecessary checks related to the envName, enhancing code clarity and maintainability. * fix(bru): prevent deletion of internal __name__ variable in deleteEnvVar method - Added a check in the deleteEnvVar method to silently ignore attempts to delete the internal __name__ variable, preserving its integrity. - Updated tests to verify that the __name__ variable remains unchanged when deleteEnvVar is called with this key. - Enhanced runtime tests to ensure compatibility with QuickJS by confirming that environment variables set with persist options are handled correctly. * feat(tests): add legacy support test for environment variable persistence - Introduced a new test suite to validate that the legacy argument for setting environment variables with persistence is still functional in version 4. - Created a fixture to simulate the legacy syntax, ensuring that the variable is correctly persisted on disk without errors. - Enhanced integration testing to confirm that the legacy behavior aligns with the current implementation, maintaining backward compatibility. * test(tests): enhance legacy environment variable persistence tests for safe and developer modes - Updated the test suite for `bru.setEnvVar` to verify that the legacy persist flag is correctly handled in both safe and developer modes. - Introduced a helper function to streamline the verification process and ensure consistent behavior across different execution contexts. - Adjusted the test logic to reset the environment state between mode switches, maintaining test integrity. - Improved hover interaction in multiple persistent variable tests to ensure reliable visibility of actions during execution. * fix(EnvironmentVariablesTable): correct change detection logic for environment variables - Updated the logic for determining changes in environment variables to compare active current and saved values instead of previously used variablesToSave and savedValues. - This change ensures accurate detection of modifications before saving, improving user feedback when no changes are present. * test(tests): enhance secret variable persistence tests for environment configurations - Updated the test suites for `bru.setEnvVar` and `bru.setGlobalEnvVar` to include interactions with the secrets tab, ensuring visibility of secret variables during various states of the environment. - Added checks to confirm that the eye toggle functionality correctly reveals the values of secret variables after setting and overwriting them. - Improved test coverage for secret variable persistence, validating that the expected values are displayed in both collection and global environment contexts.
1 parent 30b4512 commit 87f7426

171 files changed

Lines changed: 7110 additions & 990 deletions

File tree

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

packages/bruno-app/src/components/EnvironmentVariablesTable/index.js

Lines changed: 1 addition & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -445,18 +445,7 @@ const EnvironmentVariablesTable = ({
445445
const otherCurrent = namedValues.filter((variable) => !belongsToActiveTab(variable));
446446
const otherSaved = savedValues.filter((variable) => !belongsToActiveTab(variable));
447447

448-
// Compare against what's on disk: for an ephemeral overlay, that's
449-
// `persistedValue`, not the scripted value Redux is holding.
450-
const baselineForCompare = (v) => {
451-
const stripped = stripEnvVarUid(v);
452-
if (v?.ephemeral && v?.persistedValue !== undefined) {
453-
stripped.value = v.persistedValue;
454-
}
455-
return stripped;
456-
};
457-
// Compare without UIDs; only the active tab's subset decides if there's anything to save.
458-
const hasChanges
459-
= JSON.stringify(activeCurrent.map(stripEnvVarUid)) !== JSON.stringify(activeSaved.map(baselineForCompare));
448+
const hasChanges = JSON.stringify(activeCurrent.map(stripEnvVarUid)) !== JSON.stringify(activeSaved.map(stripEnvVarUid));
460449
if (!hasChanges) {
461450
toast.error('No changes to save');
462451
return;
@@ -759,11 +748,6 @@ const EnvironmentVariablesTable = ({
759748
isSecret={variable.secret}
760749
onChange={(newValue) => {
761750
formik.setFieldValue(`${actualIndex}.value`, newValue, true);
762-
// Clear ephemeral metadata when user manually edits the value
763-
if (variable.ephemeral) {
764-
formik.setFieldValue(`${actualIndex}.ephemeral`, undefined, false);
765-
formik.setFieldValue(`${actualIndex}.persistedValue`, undefined, false);
766-
}
767751
// Append a new empty row when editing value on the last row
768752
if (isLastRow) {
769753
setTimeout(() => {

packages/bruno-app/src/components/Environments/ConfirmCloseEnvironment/index.js

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -33,15 +33,15 @@ const ConfirmCloseEnvironment = ({ onCancel, onCloseWithoutSave, onSaveAndClose,
3333

3434
<div className="flex justify-between mt-6">
3535
<div>
36-
<Button color="danger" onClick={onCloseWithoutSave}>
36+
<Button color="danger" onClick={onCloseWithoutSave} data-testid="env-unsaved-close-without-save">
3737
Don't Save
3838
</Button>
3939
</div>
4040
<div className="flex gap-2">
41-
<Button size="sm" color="secondary" variant="ghost" onClick={onCancel}>
41+
<Button size="sm" color="secondary" variant="ghost" onClick={onCancel} data-testid="env-unsaved-cancel">
4242
Cancel
4343
</Button>
44-
<Button onClick={onSaveAndClose}>
44+
<Button onClick={onSaveAndClose} data-testid="env-unsaved-save-and-close">
4545
Save
4646
</Button>
4747
</div>

packages/bruno-app/src/components/MultiLineEditor/index.js

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -219,7 +219,7 @@ class MultiLineEditor extends Component {
219219
*/
220220
secretEye = (isSecret) => {
221221
return isSecret === true ? (
222-
<button className="mx-2" onClick={() => this.toggleVisibleSecret()}>
222+
<button className="mx-2" data-testid="secret-reveal-toggle" onClick={() => this.toggleVisibleSecret()}>
223223
{this.state.maskInput === true ? (
224224
<IconEyeOff size={18} strokeWidth={2} />
225225
) : (

packages/bruno-app/src/components/SingleLineEditor/index.js

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -302,7 +302,7 @@ class SingleLineEditor extends Component {
302302
*/
303303
secretEye = (isSecret) => {
304304
return isSecret === true ? (
305-
<button type="button" className="mx-2" onClick={() => this.toggleVisibleSecret()}>
305+
<button type="button" className="mx-2" data-testid="secret-reveal-toggle" onClick={() => this.toggleVisibleSecret()}>
306306
{this.state.maskInput === true ? (
307307
<IconEyeOff size={18} strokeWidth={2} />
308308
) : (

packages/bruno-app/src/providers/App/useIpcEvents.js

Lines changed: 22 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -22,10 +22,11 @@ import {
2222
runFolderEvent,
2323
runRequestEvent,
2424
scriptEnvironmentUpdateEvent,
25+
runtimeVariablesUpdateEvent,
2526
streamDataReceived,
2627
setDotEnvVariables
2728
} from 'providers/ReduxStore/slices/collections';
28-
import { collectionAddEnvFileEvent, openCollectionEvent, hydrateCollectionWithUiStateSnapshot, mergeAndPersistEnvironment } from 'providers/ReduxStore/slices/collections/actions';
29+
import { collectionAddEnvFileEvent, openCollectionEvent, hydrateCollectionWithUiStateSnapshot, persistActiveEnvironment, collectionVariablesUpdateEvent } from 'providers/ReduxStore/slices/collections/actions';
2930
import {
3031
workspaceOpenedEvent,
3132
workspaceConfigUpdatedEvent,
@@ -35,7 +36,7 @@ import { workspaceDotEnvUpdateEvent, setWorkspaceDotEnvVariables } from 'provide
3536
import toast from 'react-hot-toast';
3637
import { useDispatch, useStore } from 'react-redux';
3738
import { isElectron } from 'utils/common/platform';
38-
import { globalEnvironmentsUpdateEvent, updateGlobalEnvironments } from 'providers/ReduxStore/slices/global-environments';
39+
import { globalEnvironmentsUpdateEvent, updateGlobalEnvironments, _clearScriptGlobalEnvBaseline } from 'providers/ReduxStore/slices/global-environments';
3940
import { collectionAddOauth2CredentialsByUrl, collectionClearOauth2CredentialsByCredentialsId, updateCollectionLoadingState, collectionLoadedFromTree } from 'providers/ReduxStore/slices/collections/index';
4041
import { addLog } from 'providers/ReduxStore/slices/logs';
4142
import { loadNotifications } from 'providers/ReduxStore/slices/notifications';
@@ -201,23 +202,36 @@ const useIpcEvents = () => {
201202
}
202203
});
203204

204-
const removeScriptEnvUpdateListener = ipcRenderer.on('main:script-environment-update', (val) => {
205-
dispatch(scriptEnvironmentUpdateEvent(val));
205+
const removeRuntimeVariablesUpdateListener = ipcRenderer.on('main:runtime-variables-update', (val) => {
206+
dispatch(runtimeVariablesUpdateEvent(val));
206207
});
207208

208-
const removePersistentEnvVariablesUpdateListener = ipcRenderer.on('main:persistent-env-variables-update', (val) => {
209-
dispatch(mergeAndPersistEnvironment(val));
209+
const removeScriptEnvUpdateListener = ipcRenderer.on('main:script-environment-update', (val) => {
210+
dispatch(scriptEnvironmentUpdateEvent(val));
211+
if (val.collectionUid) {
212+
dispatch(persistActiveEnvironment(val.collectionUid, val.requestUid));
213+
}
210214
});
211215

212216
const removeGlobalEnvironmentVariablesUpdateListener = ipcRenderer.on('main:global-environment-variables-update', (val) => {
213217
dispatch(globalEnvironmentsUpdateEvent(val));
214218
});
215219

220+
const removeCollectionVariablesUpdateListener = ipcRenderer.on('main:collection-variables-update', (val) => {
221+
dispatch(collectionVariablesUpdateEvent(val));
222+
});
223+
216224
const removeCollectionRenamedListener = ipcRenderer.on('main:collection-renamed', (val) => {
217225
dispatch(collectionRenamedEvent(val));
218226
});
219227

220228
const removeRunFolderEventListener = ipcRenderer.on('main:run-folder-event', (val) => {
229+
// Folder runs reuse the workspace baseline across N requests; clear it
230+
// per request so request N's global-env update doesn't diff against
231+
// request N-1's pre-flush snapshot.
232+
if (val.type === 'testrun-started' || val.type === 'request-queued') {
233+
dispatch(_clearScriptGlobalEnvBaseline());
234+
}
221235
dispatch(runFolderEvent(val));
222236
});
223237

@@ -393,7 +407,8 @@ const useIpcEvents = () => {
393407
removeHttpStreamNewDataListener();
394408
removeHttpStreamEndListener();
395409
removeCollectionLoadingStateListener();
396-
removePersistentEnvVariablesUpdateListener();
410+
removeCollectionVariablesUpdateListener();
411+
removeRuntimeVariablesUpdateListener();
397412
removeSystemResourcesListener();
398413
gitVersionListener();
399414
removeLoadNotificationsListener();

0 commit comments

Comments
 (0)