mirror of
https://github.com/DeNNiiInc/dbgate.git
synced 2026-04-17 23:45:59 +00:00
306 lines
8.8 KiB
TypeScript
306 lines
8.8 KiB
TypeScript
import resolveApi, { resolveApiHeaders } from './resolveApi';
|
|
import { writable } from 'svelte/store';
|
|
// import { cacheClean } from './cache';
|
|
import getElectron from './getElectron';
|
|
// import socket from './socket';
|
|
import { showSnackbarError } from '../utility/snackbar';
|
|
import { handleAuthOnStartup, isOauthCallback, redirectToAdminLogin, redirectToLogin } from '../clientAuth';
|
|
import { showModal } from '../modals/modalTools';
|
|
import DatabaseLoginModal, { isDatabaseLoginVisible } from '../modals/DatabaseLoginModal.svelte';
|
|
import _ from 'lodash';
|
|
import uuidv1 from 'uuid/v1';
|
|
import { callServerPing } from './connectionsPinger';
|
|
import { batchDispatchCacheTriggers, dispatchCacheChange } from './cache';
|
|
import { isAdminPage, isOneOfPage } from './pageDefs';
|
|
import { openWebLink } from './simpleTools';
|
|
import { serializeJsTypesReplacer } from 'dbgate-tools';
|
|
|
|
export const strmid = uuidv1();
|
|
|
|
let eventSource;
|
|
let apiLogging = false;
|
|
// let cacheCleanerRegistered;
|
|
let apiDisabled = false;
|
|
const disabledOnOauth = isOauthCallback();
|
|
|
|
export const volatileConnectionMapStore = writable({});
|
|
export const volatileConnectionMapInvStore = writable({});
|
|
|
|
let volatileConnectionMapValue = {};
|
|
volatileConnectionMapStore.subscribe(value => {
|
|
volatileConnectionMapValue = value;
|
|
});
|
|
export const getVolatileConnectionMap = () => volatileConnectionMapValue;
|
|
|
|
let volatileConnectionMapInvValue = {};
|
|
volatileConnectionMapInvStore.subscribe(value => {
|
|
volatileConnectionMapInvValue = value;
|
|
});
|
|
export const getVolatileConnectionInvMap = () => volatileConnectionMapInvValue;
|
|
|
|
export function disableApi() {
|
|
apiDisabled = true;
|
|
}
|
|
|
|
export function enableApi() {
|
|
apiDisabled = false;
|
|
}
|
|
|
|
export function setVolatileConnectionRemapping(existingConnectionId, volatileConnectionId) {
|
|
volatileConnectionMapStore.update(x => ({
|
|
...x,
|
|
[existingConnectionId]: volatileConnectionId,
|
|
}));
|
|
volatileConnectionMapInvStore.update(x => ({
|
|
...x,
|
|
[volatileConnectionId]: existingConnectionId,
|
|
}));
|
|
}
|
|
|
|
export function getVolatileRemapping(conid) {
|
|
return volatileConnectionMapValue[conid] || conid;
|
|
}
|
|
|
|
export function getVolatileRemappingInv(conid) {
|
|
return volatileConnectionMapInvValue[conid] || conid;
|
|
}
|
|
|
|
export function removeVolatileMapping(conid) {
|
|
const mapped = volatileConnectionMapValue[conid];
|
|
if (mapped) {
|
|
volatileConnectionMapStore.update(x => _.omit(x, conid));
|
|
volatileConnectionMapInvStore.update(x => _.omit(x, mapped));
|
|
}
|
|
}
|
|
|
|
function wantEventSource() {
|
|
if (!eventSource) {
|
|
eventSource = new EventSource(`${resolveApi()}/stream?strmid=${strmid}`);
|
|
// eventSource.addEventListener('clean-cache', e => cacheClean(JSON.parse(e.data)));
|
|
}
|
|
}
|
|
|
|
async function processApiResponse(route, args, resp) {
|
|
// if (apiLogging) {
|
|
// console.log('<<< API RESPONSE', route, args, resp);
|
|
// }
|
|
|
|
if (resp?.missingCredentials) {
|
|
if (resp.detail.redirectToDbLogin) {
|
|
const volatile = await apiCall('connections/volatile-dblogin-from-auth', { conid: resp.detail.conid });
|
|
if (volatile) {
|
|
setVolatileConnectionRemapping(resp.detail.conid, volatile._id);
|
|
await callServerPing();
|
|
dispatchCacheChange({ key: `server-status-changed` });
|
|
batchDispatchCacheTriggers(x => x.conid == resp.detail.conid);
|
|
return null;
|
|
}
|
|
|
|
const state = `dbg-dblogin:${strmid}:${resp.detail.conid}`;
|
|
localStorage.setItem('dbloginState', state);
|
|
if (getElectron()) {
|
|
const dbloginApp = await apiCall('connections/dblogin-app', {
|
|
conid: resp.detail.conid,
|
|
state,
|
|
});
|
|
openWebLink(dbloginApp.url);
|
|
} else {
|
|
openWebLink(
|
|
`connections/dblogin-web?conid=${resp.detail.conid}&state=${encodeURIComponent(state)}&redirectUri=${
|
|
location.origin + location.pathname
|
|
}`
|
|
);
|
|
}
|
|
} else if (!isDatabaseLoginVisible()) {
|
|
showModal(DatabaseLoginModal, resp.detail);
|
|
}
|
|
return null;
|
|
// return {
|
|
// errorMessage: resp.apiErrorMessage,
|
|
// missingCredentials: true,
|
|
// };
|
|
} else if (resp?.apiErrorMessage) {
|
|
showSnackbarError('API error:' + resp?.apiErrorMessage);
|
|
return {
|
|
errorMessage: resp.apiErrorMessage,
|
|
};
|
|
}
|
|
|
|
return resp;
|
|
}
|
|
|
|
export function transformApiArgs(args) {
|
|
return _.mapValues(args, (v, k) => {
|
|
if (k == 'conid' && v && volatileConnectionMapValue[v]) return volatileConnectionMapValue[v];
|
|
if (k == 'conidArray' && _.isArray(v)) return v.map(x => volatileConnectionMapValue[x] || x);
|
|
return v;
|
|
});
|
|
}
|
|
|
|
export function transformApiArgsInv(args) {
|
|
return _.mapValues(args, (v, k) => {
|
|
if (k == 'conid' && v && volatileConnectionMapInvValue[v]) return volatileConnectionMapInvValue[v];
|
|
if (k == 'conidArray' && _.isArray(v)) return v.map(x => volatileConnectionMapInvValue[x] || x);
|
|
return v;
|
|
});
|
|
}
|
|
|
|
export async function apiCall(
|
|
route: string,
|
|
args: {} = undefined,
|
|
options: { skipDisableChecks: boolean } = undefined
|
|
) {
|
|
if (apiLogging) {
|
|
console.log('>>> API CALL', route, args);
|
|
}
|
|
if (!options?.skipDisableChecks) {
|
|
if (apiDisabled) {
|
|
console.log('API disabled!!', route);
|
|
return;
|
|
}
|
|
if (disabledOnOauth && route != 'auth/oauth-token') {
|
|
console.log('API disabled because oauth callback!!', route);
|
|
return;
|
|
}
|
|
}
|
|
|
|
args = transformApiArgs(args);
|
|
|
|
const electron = getElectron();
|
|
if (electron) {
|
|
const resp = await electron.invoke(route.replace('/', '-'), args);
|
|
return await processApiResponse(route, args, resp);
|
|
} else {
|
|
const resp = await fetch(`${resolveApi()}/${route}`, {
|
|
method: 'POST',
|
|
cache: 'no-cache',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
...resolveApiHeaders(),
|
|
},
|
|
body: JSON.stringify(args, serializeJsTypesReplacer),
|
|
});
|
|
|
|
if (resp.status == 401 && !apiDisabled) {
|
|
const page = window['dbgate_page'];
|
|
|
|
disableApi();
|
|
console.log('Disabling API', route);
|
|
if (page != 'login' && page != 'admin-login' && page != 'not-logged') {
|
|
const config = await apiCall('config/get', {}, { skipDisableChecks: true });
|
|
await handleAuthOnStartup(config);
|
|
|
|
// // unauthorized
|
|
// if (page == 'admin') {
|
|
// redirectToAdminLogin();
|
|
// } else {
|
|
// redirectToLogin();
|
|
// }
|
|
}
|
|
return;
|
|
}
|
|
|
|
const json = await resp.json();
|
|
return await processApiResponse(route, args, json);
|
|
}
|
|
}
|
|
|
|
const apiHandlers = new WeakMap();
|
|
|
|
export function apiOn(event: string, handler: Function) {
|
|
const electron = getElectron();
|
|
if (electron) {
|
|
if (!apiHandlers.has(handler)) {
|
|
const handlerProxy = (e, data) => {
|
|
if (apiLogging) {
|
|
console.log('@@@ API EVENT', event, data);
|
|
}
|
|
handler(data);
|
|
};
|
|
apiHandlers.set(handler, handlerProxy);
|
|
}
|
|
|
|
electron.addEventListener(event, apiHandlers.get(handler));
|
|
} else {
|
|
wantEventSource();
|
|
if (!apiHandlers.has(handler)) {
|
|
const handlerProxy = e => {
|
|
const json = JSON.parse(e.data);
|
|
if (apiLogging) {
|
|
console.log('@@@ API EVENT', event, json);
|
|
}
|
|
|
|
handler(json);
|
|
};
|
|
apiHandlers.set(handler, handlerProxy);
|
|
}
|
|
|
|
eventSource.addEventListener(event, apiHandlers.get(handler));
|
|
}
|
|
|
|
// if (!cacheCleanerRegistered) {
|
|
// cacheCleanerRegistered = true;
|
|
// apiOn('clean-cache', reloadTrigger => cacheClean(reloadTrigger));
|
|
// }
|
|
}
|
|
|
|
export function apiOff(event: string, handler: Function) {
|
|
const electron = getElectron();
|
|
if (apiHandlers.has(handler)) {
|
|
if (electron) {
|
|
electron.removeEventListener(event, apiHandlers.get(handler));
|
|
} else {
|
|
wantEventSource();
|
|
eventSource.removeEventListener(event, apiHandlers.get(handler));
|
|
}
|
|
}
|
|
}
|
|
|
|
export function useApiCall(route, args, defaultValue) {
|
|
const result = writable(defaultValue);
|
|
|
|
apiCall(route, args).then(resp => {
|
|
result.set(resp);
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
export function getVolatileConnections() {
|
|
return Object.values(volatileConnectionMapValue);
|
|
}
|
|
|
|
export function installNewVolatileConnectionListener() {
|
|
apiOn('got-volatile-token', async ({ savedConId, volatileConId }) => {
|
|
setVolatileConnectionRemapping(savedConId, volatileConId);
|
|
await callServerPing();
|
|
dispatchCacheChange({ key: `server-status-changed` });
|
|
batchDispatchCacheTriggers(x => x.conid == savedConId);
|
|
});
|
|
}
|
|
|
|
export function getAuthCategory(config) {
|
|
if (config.isBasicAuth) {
|
|
return 'basic';
|
|
}
|
|
if (isOneOfPage('admin', 'admin-license') && config.isAdminLoginForm) {
|
|
return 'admin';
|
|
}
|
|
if (getElectron()) {
|
|
return 'electron';
|
|
}
|
|
return 'token';
|
|
}
|
|
|
|
function enableApiLog() {
|
|
apiLogging = true;
|
|
console.log('API loggin enabled');
|
|
}
|
|
function disableApiLog() {
|
|
apiLogging = false;
|
|
console.log('API loggin disabled');
|
|
}
|
|
|
|
window['enableApiLog'] = enableApiLog;
|
|
window['disableApiLog'] = disableApiLog;
|