insomnia/packages/insomnia-app/app/models/request-version.js

115 lines
3.0 KiB
JavaScript
Raw Normal View History

// @flow
import deepEqual from 'deep-equal';
import * as models from './index';
import * as db from '../common/database';
2018-06-25 17:42:50 +00:00
import { compressObject, decompressObject } from '../common/misc';
import type { BaseModel } from './index';
import type { Request } from './request';
export const name = 'Request Version';
export const type = 'RequestVersion';
export const prefix = 'rvr';
export const canDuplicate = false;
Version Control (beta) (#1439) * VCS proof of concept underway! * Stuff * Some things * Replace deprecated Electron makeSingleInstance * Rename `window` variables so not to be confused with window object * Don't unnecessarily update request when URL does not change * Regenerate package-lock * Fix tests + ESLint * Publish - insomnia-app@1.0.49 - insomnia-cookies@0.0.12 - insomnia-httpsnippet@1.16.18 - insomnia-importers@2.0.13 - insomnia-libcurl@0.0.23 - insomnia-prettify@0.1.7 - insomnia-url@0.1.6 - insomnia-xpath@1.0.9 - insomnia-plugin-base64@1.0.6 - insomnia-plugin-cookie-jar@1.0.8 - insomnia-plugin-core-themes@1.0.5 - insomnia-plugin-default-headers@1.1.9 - insomnia-plugin-file@1.0.7 - insomnia-plugin-hash@1.0.7 - insomnia-plugin-jsonpath@1.0.12 - insomnia-plugin-now@1.0.11 - insomnia-plugin-os@1.0.13 - insomnia-plugin-prompt@1.1.9 - insomnia-plugin-request@1.0.18 - insomnia-plugin-response@1.0.16 - insomnia-plugin-uuid@1.0.10 * Broken but w/e * Some tweaks * Big refactor. Create local snapshots and push done * POC merging and a lot of improvements * Lots of work done on initial UI/UX * Fix old tests * Atomic writes and size-based batches * Update StageEntry definition once again to be better * Factor out GraphQL query logic * Merge algorithm, history modal, other minor things * Fix test * Merge, checkout, revert w/ user changes now work * Force UI to refresh when switching branches changes active request * Rough draft pull() and some cleanup * E2EE stuff and some refactoring * Add ability to share project with team and fixed tests * VCS now created in root component and better remote project handling * Remove unused definition * Publish - insomnia-account@0.0.2 - insomnia-app@1.1.1 - insomnia-cookies@0.0.14 - insomnia-httpsnippet@1.16.20 - insomnia-importers@2.0.15 - insomnia-libcurl@0.0.25 - insomnia-prettify@0.1.9 - insomnia-sync@0.0.2 - insomnia-url@0.1.8 - insomnia-xpath@1.0.11 - insomnia-plugin-base64@1.0.8 - insomnia-plugin-cookie-jar@1.0.10 - insomnia-plugin-core-themes@1.0.7 - insomnia-plugin-file@1.0.9 - insomnia-plugin-hash@1.0.9 - insomnia-plugin-jsonpath@1.0.14 - insomnia-plugin-now@1.0.13 - insomnia-plugin-os@1.0.15 - insomnia-plugin-prompt@1.1.11 - insomnia-plugin-request@1.0.20 - insomnia-plugin-response@1.0.18 - insomnia-plugin-uuid@1.0.12 * Move some deps around * Fix Flow errors * Update package.json * Fix eslint errors * Fix tests * Update deps * bootstrap insomnia-sync * TRy fixing appveyor * Try something else * Bump lerna * try powershell * Try again * Fix imports * Fixed errors * sync types refactor * Show remote projects in workspace dropdown * Improved pulling of non-local workspaces * Loading indicators and some tweaks * Clean up sync staging modal * Some sync improvements: - No longer store stage - Upgrade Electron - Sync UI/UX improvements * Fix snyc tests * Upgraded deps and hot loader tweaks (it's broken for some reason) * Fix tests * Branches dialog, network refactoring, some tweaks * Fixed merging when other branch is empty * A bunch of small fixes from real testing * Fixed pull merge logic * Fix tests * Some bug fixes * A few small tweaks * Conflict resolution and other improvements * Fix tests * Add revert changes * Deal with duplicate projects per workspace * Some tweaks and accessibility improvements * Tooltip accessibility * Fix API endpoint * Fix tests * Remove jest dep from insomnia-importers
2019-04-18 00:50:03 +00:00
export const canSync = false;
type BaseRequestVersion = {
compressedRequest: string | null,
};
export type RequestVersion = BaseModel & BaseRequestVersion;
const FIELDS_TO_IGNORE = [
'_id',
'type',
'created',
'modified',
'metaSortKey',
'description',
'parentId',
'name',
];
2018-06-25 17:42:50 +00:00
export function init() {
return {
compressedRequest: null,
};
}
export function migrate(doc: RequestVersion): RequestVersion {
return doc;
}
export function getById(id: string): Promise<RequestVersion | null> {
return db.get(type, id);
}
export async function create(request: Request): Promise<RequestVersion> {
if (!request.type === models.request.type) {
2018-10-17 16:42:33 +00:00
throw new Error(`New ${type} was not given a valid ${models.request.type} instance`);
}
const parentId = request._id;
const latestRequestVersion: RequestVersion | null = await getLatestByParentId(parentId);
const latestRequest = latestRequestVersion
? decompressObject(latestRequestVersion.compressedRequest)
: null;
const hasChanged = _diffRequests(latestRequest, request);
console.log('HAS CHANGED', { latestRequestVersion, request });
if (hasChanged || !latestRequestVersion) {
// Create a new version if the request has been modified
const compressedRequest = compressObject(request);
2018-06-25 17:42:50 +00:00
return db.docCreate(type, { parentId, compressedRequest });
} else {
// Re-use the latest version if not modified since
return latestRequestVersion;
}
}
export function getLatestByParentId(parentId: string): Promise<RequestVersion | null> {
2018-06-25 17:42:50 +00:00
return db.getMostRecentlyModified(type, { parentId });
}
export async function restore(requestVersionId: string): Promise<Request | null> {
const requestVersion = await getById(requestVersionId);
// Older responses won't have versions saved with them
if (!requestVersion) {
return null;
}
const requestPatch = decompressObject(requestVersion.compressedRequest);
const originalRequest: Request | null = await models.request.getById(requestPatch._id);
if (!originalRequest) {
return null;
}
// Only restore fields that aren't blacklisted
for (const field of FIELDS_TO_IGNORE) {
delete requestPatch[field];
}
return models.request.update(originalRequest, requestPatch);
}
function _diffRequests(rOld: Request | null, rNew: Request): boolean {
if (!rOld) {
return true;
}
for (const key of Object.keys(rOld)) {
// Skip fields that aren't useful
if (FIELDS_TO_IGNORE.includes(key)) {
continue;
}
if (!deepEqual(rOld[key], rNew[key])) {
return true;
}
}
return false;
}