mirror of
https://github.com/Kong/insomnia
synced 2024-11-08 14:49:53 +00:00
0a616fba6b
* 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
351 lines
9.0 KiB
JavaScript
351 lines
9.0 KiB
JavaScript
// @flow
|
|
import * as electron from 'electron';
|
|
import { Readable, Writable } from 'stream';
|
|
import fuzzysort from 'fuzzysort';
|
|
import uuid from 'uuid';
|
|
import zlib from 'zlib';
|
|
import { join as pathJoin } from 'path';
|
|
import { DEBOUNCE_MILLIS } from './constants';
|
|
|
|
const ESCAPE_REGEX_MATCH = /[-[\]/{}()*+?.\\^$|]/g;
|
|
|
|
type Header = {
|
|
name: string,
|
|
value: string,
|
|
};
|
|
|
|
type Parameter = {
|
|
name: string,
|
|
value: string,
|
|
};
|
|
|
|
export function filterParameters<T: Parameter>(parameters: Array<T>, name: string): Array<T> {
|
|
if (!Array.isArray(parameters) || !name) {
|
|
return [];
|
|
}
|
|
|
|
return parameters.filter(h => (!h || !h.name ? false : h.name === name));
|
|
}
|
|
|
|
export function filterHeaders<T: Header>(headers: Array<T>, name: string): Array<T> {
|
|
if (!Array.isArray(headers) || !name) {
|
|
return [];
|
|
}
|
|
|
|
return headers.filter(h => {
|
|
if (!h || !h.name) {
|
|
return false;
|
|
} else {
|
|
return h.name.toLowerCase() === name.toLowerCase();
|
|
}
|
|
});
|
|
}
|
|
|
|
export function hasContentTypeHeader<T: Header>(headers: Array<T>): boolean {
|
|
return filterHeaders(headers, 'content-type').length > 0;
|
|
}
|
|
|
|
export function hasContentLengthHeader<T: Header>(headers: Array<T>): boolean {
|
|
return filterHeaders(headers, 'content-length').length > 0;
|
|
}
|
|
|
|
export function hasAuthHeader<T: Header>(headers: Array<T>): boolean {
|
|
return filterHeaders(headers, 'authorization').length > 0;
|
|
}
|
|
|
|
export function hasAcceptHeader<T: Header>(headers: Array<T>): boolean {
|
|
return filterHeaders(headers, 'accept').length > 0;
|
|
}
|
|
|
|
export function hasUserAgentHeader<T: Header>(headers: Array<T>): boolean {
|
|
return filterHeaders(headers, 'user-agent').length > 0;
|
|
}
|
|
|
|
export function hasAcceptEncodingHeader<T: Header>(headers: Array<T>): boolean {
|
|
return filterHeaders(headers, 'accept-encoding').length > 0;
|
|
}
|
|
|
|
export function getSetCookieHeaders<T: Header>(headers: Array<T>): Array<T> {
|
|
return filterHeaders(headers, 'set-cookie');
|
|
}
|
|
|
|
export function getLocationHeader<T: Header>(headers: Array<T>): T | null {
|
|
const matches = filterHeaders(headers, 'location');
|
|
return matches.length ? matches[0] : null;
|
|
}
|
|
|
|
export function getContentTypeHeader<T: Header>(headers: Array<T>): T | null {
|
|
const matches = filterHeaders(headers, 'content-type');
|
|
return matches.length ? matches[0] : null;
|
|
}
|
|
|
|
export function getHostHeader<T: Header>(headers: Array<T>): T | null {
|
|
const matches = filterHeaders(headers, 'host');
|
|
return matches.length ? matches[0] : null;
|
|
}
|
|
|
|
export function getContentDispositionHeader<T: Header>(headers: Array<T>): T | null {
|
|
const matches = filterHeaders(headers, 'content-disposition');
|
|
return matches.length ? matches[0] : null;
|
|
}
|
|
|
|
export function getContentLengthHeader<T: Header>(headers: Array<T>): T | null {
|
|
const matches = filterHeaders(headers, 'content-length');
|
|
return matches.length ? matches[0] : null;
|
|
}
|
|
|
|
/**
|
|
* Generate an ID of the format "<MODEL_NAME>_<TIMESTAMP><RANDOM>"
|
|
* @param prefix
|
|
* @returns {string}
|
|
*/
|
|
export function generateId(prefix: string): string {
|
|
const id = uuid.v4().replace(/-/g, '');
|
|
|
|
if (prefix) {
|
|
return `${prefix}_${id}`;
|
|
} else {
|
|
return id;
|
|
}
|
|
}
|
|
|
|
export function delay(milliseconds: number = DEBOUNCE_MILLIS): Promise<void> {
|
|
return new Promise(resolve => setTimeout(resolve, milliseconds));
|
|
}
|
|
|
|
export function removeVowels(str: string): string {
|
|
return str.replace(/[aeiouyAEIOUY]/g, '');
|
|
}
|
|
|
|
export function keyedDebounce(callback: Function, millis: number = DEBOUNCE_MILLIS): Function {
|
|
let timeout;
|
|
let results = {};
|
|
|
|
return function(key, ...args) {
|
|
results[key] = args;
|
|
|
|
clearTimeout(timeout);
|
|
timeout = setTimeout(() => {
|
|
if (!Object.keys(results).length) {
|
|
return;
|
|
}
|
|
|
|
callback(results);
|
|
results = {};
|
|
}, millis);
|
|
};
|
|
}
|
|
|
|
export function debounce(callback: Function, millis: number = DEBOUNCE_MILLIS): Function {
|
|
// For regular debounce, just use a keyed debounce with a fixed key
|
|
return keyedDebounce(results => {
|
|
callback.apply(null, results['__key__']);
|
|
}, millis).bind(null, '__key__');
|
|
}
|
|
|
|
export function describeByteSize(bytes: number, long: boolean = false): string {
|
|
bytes = Math.round(bytes * 10) / 10;
|
|
let size;
|
|
|
|
// NOTE: We multiply these by 2 so we don't end up with
|
|
// values like 0 GB
|
|
|
|
let unit = long ? 'bytes' : 'B';
|
|
if (bytes < 1024 * 2) {
|
|
size = bytes;
|
|
unit = long ? 'bytes' : 'B';
|
|
} else if (bytes < 1024 * 1024 * 2) {
|
|
size = bytes / 1024;
|
|
unit = long ? 'kilobytes' : 'KB';
|
|
} else if (bytes < 1024 * 1024 * 1024 * 2) {
|
|
size = bytes / 1024 / 1024;
|
|
unit = long ? 'megabytes' : 'MB';
|
|
} else {
|
|
size = bytes / 1024 / 1024 / 1024;
|
|
unit = long ? 'gigabytes' : 'GB';
|
|
}
|
|
|
|
const rounded = Math.round(size * 10) / 10;
|
|
return `${rounded} ${unit}`;
|
|
}
|
|
|
|
export function nullFn(): void {
|
|
// Do nothing
|
|
}
|
|
|
|
export function preventDefault(e: Event): void {
|
|
e.preventDefault();
|
|
}
|
|
|
|
export function clickLink(href: string): void {
|
|
electron.shell.openExternal(href);
|
|
}
|
|
|
|
export function fnOrString(v: string | Function, ...args: Array<any>) {
|
|
if (typeof v === 'string') {
|
|
return v;
|
|
} else {
|
|
return v(...args);
|
|
}
|
|
}
|
|
|
|
export function compressObject(obj: any): string {
|
|
const compressed = zlib.gzipSync(JSON.stringify(obj));
|
|
return compressed.toString('base64');
|
|
}
|
|
|
|
export function decompressObject(input: string): any {
|
|
const jsonBuffer = zlib.gunzipSync(Buffer.from(input, 'base64'));
|
|
return JSON.parse(jsonBuffer.toString('utf8'));
|
|
}
|
|
|
|
export function resolveHomePath(p: string): string {
|
|
if (p.indexOf('~/') === 0) {
|
|
return pathJoin(process.env.HOME || '/', p.slice(1));
|
|
} else {
|
|
return p;
|
|
}
|
|
}
|
|
|
|
export function jsonParseOr(str: string, fallback: any): any {
|
|
try {
|
|
return JSON.parse(str);
|
|
} catch (err) {
|
|
return fallback;
|
|
}
|
|
}
|
|
|
|
export function escapeHTML(unsafeText: string): string {
|
|
let div = document.createElement('div');
|
|
div.innerText = unsafeText;
|
|
return div.innerHTML;
|
|
}
|
|
|
|
/**
|
|
* Escape a dynamic string for use inside of a regular expression
|
|
* @param str - string to escape
|
|
* @returns {string} escaped string
|
|
*/
|
|
export function escapeRegex(str: string): string {
|
|
return str.replace(ESCAPE_REGEX_MATCH, '\\$&');
|
|
}
|
|
|
|
export function fuzzyMatch(
|
|
searchString: string,
|
|
text: string,
|
|
options: { splitSpace?: boolean, loose?: boolean } = {},
|
|
): null | { score: number, indexes: Array<number> } {
|
|
return fuzzyMatchAll(searchString, [text], options);
|
|
}
|
|
|
|
export function fuzzyMatchAll(
|
|
searchString: string,
|
|
allText: Array<string>,
|
|
options: { splitSpace?: boolean, loose?: boolean } = {},
|
|
): null | { score: number, indexes: Array<number> } {
|
|
if (!searchString || !searchString.trim()) {
|
|
return null;
|
|
}
|
|
|
|
const words = searchString.split(' ').filter(w => w.trim());
|
|
const terms = options.splitSpace ? [...words, searchString] : [searchString];
|
|
|
|
let maxScore = null;
|
|
let indexes = [];
|
|
let termsMatched = 0;
|
|
for (const term of terms) {
|
|
let matchedTerm = false;
|
|
for (const text of allText.filter(t => !t || t.trim())) {
|
|
const result = fuzzysort.single(term, text);
|
|
if (!result) {
|
|
continue;
|
|
}
|
|
|
|
// Don't match garbage
|
|
if (result.score < -8000) {
|
|
continue;
|
|
}
|
|
|
|
if (maxScore === null || result.score > maxScore) {
|
|
maxScore = result.score;
|
|
}
|
|
|
|
indexes = [...indexes, ...result.indexes];
|
|
matchedTerm = true;
|
|
}
|
|
|
|
if (matchedTerm) {
|
|
termsMatched++;
|
|
}
|
|
}
|
|
|
|
// Make sure we match all provided terms except the last (full) one
|
|
if (!options.loose && termsMatched < terms.length - 1) {
|
|
return null;
|
|
}
|
|
|
|
if (maxScore === null) {
|
|
return null;
|
|
}
|
|
|
|
return { score: maxScore, indexes, target: allText.join(' ') };
|
|
}
|
|
|
|
export function getViewportSize(): string | null {
|
|
const { BrowserWindow } = electron.remote || electron;
|
|
const w = BrowserWindow.getFocusedWindow() || BrowserWindow.getAllWindows()[0];
|
|
|
|
if (w) {
|
|
const { width, height } = w.getContentBounds();
|
|
return `${width}x${height}`;
|
|
} else {
|
|
// No windows open
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export function getScreenResolution(): string {
|
|
const { screen } = electron.remote || electron;
|
|
const { width, height } = screen.getPrimaryDisplay().workAreaSize;
|
|
return `${width}x${height}`;
|
|
}
|
|
|
|
export function getUserLanguage(): string {
|
|
const { app } = electron.remote || electron;
|
|
return app.getLocale();
|
|
}
|
|
|
|
export async function waitForStreamToFinish(s: Readable | Writable): Promise<void> {
|
|
return new Promise(resolve => {
|
|
if ((s: any)._readableState && (s: any)._readableState.finished) {
|
|
return resolve();
|
|
}
|
|
|
|
if ((s: any)._writableState && (s: any)._writableState.finished) {
|
|
return resolve();
|
|
}
|
|
|
|
s.on('close', () => {
|
|
resolve();
|
|
});
|
|
|
|
s.on('error', () => {
|
|
resolve();
|
|
});
|
|
});
|
|
}
|
|
|
|
export function getDataDirectory(): string {
|
|
const { app } = electron.remote || electron;
|
|
return process.env.INSOMNIA_DATA_PATH || app.getPath('userData');
|
|
}
|
|
|
|
export function chunkArray<T>(arr: Array<T>, chunkSize: number): Array<Array<T>> {
|
|
const chunks = [];
|
|
for (let i = 0, j = arr.length; i < j; i += chunkSize) {
|
|
chunks.push(arr.slice(i, i + chunkSize));
|
|
}
|
|
|
|
return chunks;
|
|
}
|