mirror of
https://github.com/Kong/insomnia
synced 2024-11-08 14:49:53 +00:00
339 lines
7.5 KiB
JavaScript
339 lines
7.5 KiB
JavaScript
import electron from 'electron';
|
|
import * as fsPath from 'path';
|
|
import * as fs from 'fs';
|
|
|
|
import * as methods from '../lib/constants';
|
|
import {generateId} from './util';
|
|
import {PREVIEW_MODE_SOURCE} from '../lib/previewModes';
|
|
import {CONTENT_TYPE_TEXT} from '../lib/contentTypes';
|
|
import {DB_PERSIST_INTERVAL, DEFAULT_SIDEBAR_WIDTH} from '../lib/constants';
|
|
|
|
export const TYPE_WORKSPACE = 'Workspace';
|
|
export const TYPE_REQUEST_GROUP = 'RequestGroup';
|
|
export const TYPE_REQUEST = 'Request';
|
|
export const TYPE_RESPONSE = 'Response';
|
|
const TYPES = [
|
|
TYPE_WORKSPACE,
|
|
TYPE_REQUEST_GROUP,
|
|
TYPE_REQUEST,
|
|
TYPE_RESPONSE
|
|
];
|
|
|
|
let db = null;
|
|
global.db = db;
|
|
|
|
function getDBFilePath () {
|
|
const basePath = electron.remote.app.getPath('userData');
|
|
return fsPath.join(basePath, 'insomnia.db.json');
|
|
}
|
|
|
|
/**
|
|
* Initialize the database. This should be called once on app start.
|
|
* @returns {Promise}
|
|
*/
|
|
let initialized = false;
|
|
export function initDB () {
|
|
// Only init once
|
|
if (initialized) {
|
|
return new Promise(resolve => resolve());
|
|
}
|
|
|
|
return new Promise(resolve => {
|
|
const dbPath = getDBFilePath();
|
|
|
|
db = {
|
|
created: Date.now(),
|
|
entities: {}
|
|
};
|
|
|
|
for (let i = 0; i < TYPES.length; i++) {
|
|
db.entities[TYPES[i]] = {};
|
|
}
|
|
|
|
fs.readFile(getDBFilePath(), 'utf8', (err, text) => {
|
|
if (!err) {
|
|
// TODO: Better error handling
|
|
console.log('-- Restored DB from file --');
|
|
try {
|
|
Object.assign(db, JSON.parse(text));
|
|
} catch (e) {
|
|
console.error('Failed to parse DB file', e);
|
|
}
|
|
}
|
|
|
|
// Add listeners to do persistence
|
|
|
|
let timeout = null;
|
|
onChange('DB_WRITER', () => {
|
|
clearTimeout(timeout);
|
|
timeout = setTimeout(() => {
|
|
|
|
// First, write to a tmp file, then overwrite the old one. This
|
|
// prevents getting a corru
|
|
const filePath = getDBFilePath();
|
|
const tmpFilePath = `${filePath}.tmp`;
|
|
|
|
fs.writeFile(tmpFilePath, JSON.stringify(db, null, 2), err => {
|
|
if (err) {
|
|
console.error('Failed to write DB to file', err);
|
|
} else {
|
|
fs.renameSync(tmpFilePath, filePath);
|
|
console.log('-- Persisted DB --');
|
|
}
|
|
})
|
|
}, DB_PERSIST_INTERVAL)
|
|
});
|
|
|
|
// Done
|
|
|
|
initialized = true;
|
|
console.log(`-- Initialize DB at ${dbPath} --`);
|
|
resolve();
|
|
});
|
|
})
|
|
}
|
|
|
|
let changeListeners = {};
|
|
|
|
export function onChange (id, callback) {
|
|
console.log(`-- Added DB Listener ${id} -- `);
|
|
changeListeners[id] = callback;
|
|
}
|
|
|
|
export function offChange (id) {
|
|
console.log(`-- Removed DB Listener ${id} -- `);
|
|
delete changeListeners[id];
|
|
}
|
|
|
|
export function get (type, id) {
|
|
const doc = db.entities[type][id];
|
|
return new Promise(resolve => resolve(doc));
|
|
}
|
|
|
|
function all (type) {
|
|
let docs = [];
|
|
const ids = Object.keys(db.entities[type]);
|
|
for (let i = 0; i < ids.length; i++) {
|
|
docs.push(db.entities[type][ids[i]]);
|
|
}
|
|
|
|
return new Promise(resolve => resolve(docs));
|
|
}
|
|
|
|
function removeWhere (type, key, value) {
|
|
const ids = Object.keys(db.entities[type]);
|
|
let docs = [];
|
|
|
|
for (let i = 0; i < ids.length; i++) {
|
|
const doc = db.entities[type][ids[i]];
|
|
if (doc[key] === value) {
|
|
remove(doc);
|
|
}
|
|
}
|
|
|
|
return new Promise(resolve => resolve(docs));
|
|
}
|
|
|
|
function insert (doc) {
|
|
db.entities[doc.type][doc._id] = doc;
|
|
|
|
Object.keys(changeListeners).map(k => changeListeners[k]('insert', doc));
|
|
return new Promise(resolve => resolve(doc));
|
|
}
|
|
|
|
function update (doc) {
|
|
db.entities[doc.type][doc._id] = doc;
|
|
|
|
Object.keys(changeListeners).map(k => changeListeners[k]('update', doc));
|
|
return new Promise(resolve => resolve(doc));
|
|
}
|
|
|
|
function remove (doc) {
|
|
delete db.entities[doc.type][doc._id];
|
|
|
|
// Also remove children
|
|
TYPES.map(type => removeWhere(type, 'parentId', doc._id));
|
|
|
|
Object.keys(changeListeners).map(k => changeListeners[k]('remove', doc));
|
|
new Promise(resolve => resolve(doc));
|
|
}
|
|
|
|
|
|
// ~~~~~~~~~~~~~~~~~~~ //
|
|
// DEFAULT MODEL STUFF //
|
|
// ~~~~~~~~~~~~~~~~~~~ //
|
|
|
|
function docUpdate (originalDoc, patch = {}) {
|
|
const doc = Object.assign(
|
|
{},
|
|
originalDoc,
|
|
patch,
|
|
{modified: Date.now()}
|
|
);
|
|
|
|
// Fake a promise
|
|
const finalDoc = update(doc);
|
|
return new Promise(resolve => resolve(finalDoc));
|
|
}
|
|
|
|
function docCreate (type, idPrefix, defaults, patch = {}) {
|
|
const baseDefaults = {
|
|
parentId: null
|
|
};
|
|
|
|
const doc = Object.assign(
|
|
baseDefaults,
|
|
defaults,
|
|
patch,
|
|
|
|
// Required Generated Fields
|
|
{
|
|
_id: generateId(idPrefix),
|
|
$loki: undefined,
|
|
meta: undefined,
|
|
type: type,
|
|
created: Date.now(),
|
|
modified: Date.now()
|
|
}
|
|
);
|
|
|
|
// Fake a promise
|
|
return insert(doc);
|
|
}
|
|
|
|
// ~~~~~~~ //
|
|
// REQUEST //
|
|
// ~~~~~~~ //
|
|
|
|
export function requestCreateAndActivate (workspace, patch = {}) {
|
|
return requestCreate(patch).then(r => {
|
|
workspaceUpdate(workspace, {activeRequestId: r._id});
|
|
})
|
|
}
|
|
|
|
export function requestCopyAndActivate (workspace, request) {
|
|
return requestCopy(request).then(r => {
|
|
workspaceUpdate(workspace, {activeRequestId: r._id});
|
|
})
|
|
}
|
|
|
|
export function requestCreate (patch = {}) {
|
|
return docCreate(TYPE_REQUEST, 'req', {
|
|
url: '',
|
|
name: 'New Request',
|
|
method: methods.METHOD_GET,
|
|
previewMode: PREVIEW_MODE_SOURCE,
|
|
contentType: CONTENT_TYPE_TEXT,
|
|
body: '',
|
|
params: [],
|
|
headers: [],
|
|
authentication: {}
|
|
}, patch);
|
|
}
|
|
|
|
export function requestById (id) {
|
|
return get(TYPE_REQUEST, id);
|
|
}
|
|
|
|
export function requestUpdate (request, patch) {
|
|
return docUpdate(request, patch);
|
|
}
|
|
|
|
export function requestCopy (request) {
|
|
const name = `${request.name} (Copy)`;
|
|
return requestCreate(Object.assign({}, request, {name}));
|
|
}
|
|
|
|
export function requestRemove (request) {
|
|
return remove(request);
|
|
}
|
|
|
|
export function requestAll () {
|
|
return all(TYPE_REQUEST);
|
|
}
|
|
|
|
|
|
// ~~~~~~~~~~~~~ //
|
|
// REQUEST GROUP //
|
|
// ~~~~~~~~~~~~~ //
|
|
|
|
export function requestGroupCreate (patch = {}) {
|
|
return docCreate(TYPE_REQUEST_GROUP, 'grp', {
|
|
collapsed: false,
|
|
name: 'New Request Group',
|
|
environment: {}
|
|
}, patch);
|
|
}
|
|
|
|
export function requestGroupUpdate (requestGroup, patch) {
|
|
return docUpdate(requestGroup, patch);
|
|
}
|
|
|
|
export function requestGroupById (id) {
|
|
return get(TYPE_REQUEST_GROUP, id);
|
|
}
|
|
|
|
export function requestGroupRemove (requestGroup) {
|
|
return remove(requestGroup);
|
|
}
|
|
|
|
export function requestGroupAll () {
|
|
return all(TYPE_REQUEST_GROUP);
|
|
}
|
|
|
|
// ~~~~~~~~ //
|
|
// RESPONSE //
|
|
// ~~~~~~~~ //
|
|
|
|
export function responseCreate (patch = {}) {
|
|
return docCreate(TYPE_RESPONSE, 'res', {
|
|
statusCode: 0,
|
|
statusMessage: '',
|
|
contentType: 'text/plain',
|
|
url: '',
|
|
bytes: 0,
|
|
millis: 0,
|
|
headers: [],
|
|
body: '',
|
|
error: ''
|
|
}, patch);
|
|
}
|
|
|
|
export function responseAll () {
|
|
return all(TYPE_RESPONSE);
|
|
}
|
|
|
|
|
|
// ~~~~~~~~~ //
|
|
// WORKSPACE //
|
|
// ~~~~~~~~~ //
|
|
|
|
export function workspaceCreate (patch = {}) {
|
|
return docCreate(TYPE_WORKSPACE, 'wrk', {
|
|
name: 'New Workspace',
|
|
activeRequestId: null,
|
|
environments: [],
|
|
sidebarWidth: DEFAULT_SIDEBAR_WIDTH,
|
|
filter: ''
|
|
}, patch);
|
|
}
|
|
|
|
export function workspaceAll () {
|
|
return all(TYPE_WORKSPACE).then(workspaces => {
|
|
if (workspaces.length === 0) {
|
|
workspaceCreate({name: 'Insomnia'});
|
|
return workspaceAll();
|
|
} else {
|
|
return new Promise(resolve => resolve(workspaces))
|
|
}
|
|
});
|
|
}
|
|
|
|
export function workspaceUpdate (workspace, patch) {
|
|
return docUpdate(workspace, patch);
|
|
}
|
|
|
|
export function workspaceRemove (workspace) {
|
|
return remove(workspace);
|
|
}
|