diff --git a/package.json b/package.json index f39e63e8a..e37c95833 100644 --- a/package.json +++ b/package.json @@ -16,6 +16,7 @@ "start:app:debug:ssh": "cd app && cross-env DEBUG=ssh yarn start", "start:api:portal": "yarn workspace dbgate-api start:portal", "start:api:singledb": "yarn workspace dbgate-api start:singledb", + "start:api:auth": "yarn workspace dbgate-api start:auth", "start:web": "yarn workspace dbgate-web dev", "start:sqltree": "yarn workspace dbgate-sqltree start", "start:tools": "yarn workspace dbgate-tools start", diff --git a/packages/api/env/auth/.gitignore b/packages/api/env/auth/.gitignore new file mode 100644 index 000000000..2eea525d8 --- /dev/null +++ b/packages/api/env/auth/.gitignore @@ -0,0 +1 @@ +.env \ No newline at end of file diff --git a/packages/api/package.json b/packages/api/package.json index 8190e5805..236d4211c 100644 --- a/packages/api/package.json +++ b/packages/api/package.json @@ -17,6 +17,7 @@ "dbgate" ], "dependencies": { + "activedirectory2": "^2.1.0", "async-lock": "^1.2.4", "axios": "^0.21.1", "body-parser": "^1.19.0", @@ -42,6 +43,7 @@ "is-electron": "^2.2.1", "js-yaml": "^4.1.0", "json-stable-stringify": "^1.0.1", + "jsonwebtoken": "^8.5.1", "line-reader": "^0.4.0", "lodash": "^4.17.21", "ncp": "^2.0.0", @@ -57,6 +59,7 @@ "start": "env-cmd node src/index.js --listen-api", "start:portal": "env-cmd -f env/portal/.env node src/index.js --listen-api", "start:singledb": "env-cmd -f env/singledb/.env node src/index.js --listen-api", + "start:auth": "env-cmd -f env/auth/.env node src/index.js --listen-api", "start:filedb": "env-cmd node src/index.js /home/jena/test/chinook/Chinook.db --listen-api", "start:singleconn": "env-cmd node src/index.js --server localhost --user root --port 3307 --engine mysql@dbgate-plugin-mysql --password test --listen-api", "ts": "tsc", diff --git a/packages/api/src/controllers/auth.js b/packages/api/src/controllers/auth.js new file mode 100644 index 000000000..d3db8660c --- /dev/null +++ b/packages/api/src/controllers/auth.js @@ -0,0 +1,142 @@ +const axios = require('axios'); +const jwt = require('jsonwebtoken'); +const getExpressPath = require('../utility/getExpressPath'); +const uuidv1 = require('uuid/v1'); +const { getLogins } = require('../utility/hasPermission'); +const AD = require('activedirectory2').promiseWrapper; + +const tokenSecret = uuidv1(); + +function shouldAuthorizeApi() { + const logins = getLogins(); + return !!process.env.OAUTH_AUTH || !!process.env.AD_URL || (!!logins && !process.env.BASIC_AUTH); +} + +function getTokenLifetime() { + return process.env.TOKEN_LIFETIME || '1d'; +} + +function unauthorizedResponse(req, res, text) { + // if (req.path == getExpressPath('/config/get-settings')) { + // return res.json({}); + // } + // if (req.path == getExpressPath('/connections/list')) { + // return res.json([]); + // } + return res.sendStatus(401).send(text); +} + +function authMiddleware(req, res, next) { + const SKIP_AUTH_PATHS = ['/config/get', '/auth/oauth-token', '/auth/login', '/stream']; + + if (!shouldAuthorizeApi()) { + return next(); + } + let skipAuth = !!SKIP_AUTH_PATHS.find(x => req.path == getExpressPath(x)); + + const authHeader = req.headers.authorization; + if (!authHeader) { + if (skipAuth) { + return next(); + } + return unauthorizedResponse(req, res, 'missing authorization header'); + } + const token = authHeader.split(' ')[1]; + try { + const decoded = jwt.verify(token, tokenSecret); + req.user = decoded; + return next(); + } catch (err) { + if (skipAuth) { + return next(); + } + + console.log('Sending invalid token error', err.message); + + return unauthorizedResponse(req, res, 'invalid token'); + } +} + +module.exports = { + oauthToken_meta: true, + async oauthToken(params) { + const { redirectUri, code } = params; + + const resp = await axios.default.post( + `${process.env.OAUTH_TOKEN}`, + `grant_type=authorization_code&code=${encodeURIComponent(code)}&redirect_uri=${encodeURIComponent( + redirectUri + )}&client_id=${process.env.OAUTH_CLIENT_ID}&client_secret=${process.env.OAUTH_CLIENT_SECRET}` + ); + + const { access_token, refresh_token } = resp.data; + + const payload = jwt.decode(access_token); + + console.log('User payload returned from OAUTH:', payload); + + const login = process.env.OAUTH_LOGIN_FIELD ? payload[process.env.OAUTH_LOGIN_FIELD] : 'oauth'; + + if ( + process.env.OAUTH_ALLOWED_LOGINS && + !process.env.OAUTH_ALLOWED_LOGINS.split(',').find(x => x.toLowerCase().trim() == login.toLowerCase().trim()) + ) { + return { error: `Username ${login} not allowed to log in` }; + } + if (access_token) { + return { + accessToken: jwt.sign({ login }, tokenSecret, { expiresIn: getTokenLifetime() }), + }; + } + + return { error: 'Token not found' }; + }, + login_meta: true, + async login(params) { + const { login, password } = params; + + if (process.env.AD_URL) { + const adConfig = { + url: process.env.AD_URL, + baseDN: process.env.AD_BASEDN, + username: process.env.AD_USERNAME, + password: process.env.AD_PASSOWRD, + }; + const ad = new AD(adConfig); + try { + const res = await ad.authenticate(login, password); + if (!res) { + return { error: 'Login failed' }; + } + if ( + process.env.AD_ALLOWED_LOGINS && + !process.env.AD_ALLOWED_LOGINS.split(',').find(x => x.toLowerCase().trim() == login.toLowerCase().trim()) + ) { + return { error: `Username ${login} not allowed to log in` }; + } + return { + accessToken: jwt.sign({ login }, tokenSecret, { expiresIn: getTokenLifetime() }), + }; + } catch (err) { + console.log('Failed active directory authentization', err.message); + return { + error: err.message, + }; + } + } + + const logins = getLogins(); + if (!logins) { + return { error: 'Logins not configured' }; + } + if (logins.find(x => x.login == login)?.password == password) { + return { + accessToken: jwt.sign({ login }, tokenSecret, { expiresIn: getTokenLifetime() }), + }; + } + return { error: 'Invalid credentials' }; + }, + + authMiddleware, + shouldAuthorizeApi, +}; diff --git a/packages/api/src/controllers/config.js b/packages/api/src/controllers/config.js index 34e3e3c8b..dbfe6ef05 100644 --- a/packages/api/src/controllers/config.js +++ b/packages/api/src/controllers/config.js @@ -28,7 +28,7 @@ module.exports = { get_meta: true, async get(_params, req) { const logins = getLogins(); - const login = logins ? logins.find(x => x.login == (req.auth && req.auth.user)) : null; + const login = req.user ? req.user.login : logins ? logins.find(x => x.login == (req.auth && req.auth.user)) : null; const permissions = login ? login.permissions : process.env.PERMISSIONS; return { @@ -40,6 +40,9 @@ module.exports = { isDocker: platformInfo.isDocker, permissions, login, + oauth: process.env.OAUTH_AUTH, + oauthLogout: process.env.OAUTH_LOGOUT, + isLoginForm: !!process.env.AD_URL || (!!logins && !process.env.BASIC_AUTH), ...currentVersion, }; }, diff --git a/packages/api/src/main.js b/packages/api/src/main.js index eec93c02a..2edf4ca8b 100644 --- a/packages/api/src/main.js +++ b/packages/api/src/main.js @@ -20,6 +20,7 @@ const jsldata = require('./controllers/jsldata'); const config = require('./controllers/config'); const archive = require('./controllers/archive'); const apps = require('./controllers/apps'); +const auth = require('./controllers/auth'); const uploads = require('./controllers/uploads'); const plugins = require('./controllers/plugins'); const files = require('./controllers/files'); @@ -41,7 +42,7 @@ function start() { const server = http.createServer(app); const logins = getLogins(); - if (logins) { + if (logins && process.env.BASIC_AUTH) { app.use( basicAuth({ users: _.fromPairs(logins.map(x => [x.login, x.password])), @@ -53,6 +54,10 @@ function start() { app.use(cors()); + if (auth.shouldAuthorizeApi()) { + app.use(auth.authMiddleware); + } + app.get(getExpressPath('/stream'), async function (req, res) { res.set({ 'Cache-Control': 'no-cache', @@ -157,6 +162,7 @@ function useAllControllers(app, electron) { useController(app, electron, '/scheduler', scheduler); useController(app, electron, '/query-history', queryHistory); useController(app, electron, '/apps', apps); + useController(app, electron, '/auth', auth); } function setElectronSender(electronSender) { diff --git a/packages/web/src/App.svelte b/packages/web/src/App.svelte index 6938737e8..c8fd1ad07 100644 --- a/packages/web/src/App.svelte +++ b/packages/web/src/App.svelte @@ -20,6 +20,7 @@ import getElectron from './utility/getElectron'; import AppStartInfo from './widgets/AppStartInfo.svelte'; import SettingsListener from './utility/SettingsListener.svelte'; + import { handleAuthOnStartup, handleOauthCallback } from './clientAuth'; let loadedApi = false; let loadedPlugins = false; @@ -33,9 +34,11 @@ try { // console.log('************** LOADING API'); + const config = await getConfig(); + await handleAuthOnStartup(config); + const connections = await apiCall('connections/list'); const settings = await getSettings(); - const config = await getConfig(); const apps = await getUsedApps(); loadedApi = settings && connections && config && apps; diff --git a/packages/web/src/LoginPage.svelte b/packages/web/src/LoginPage.svelte new file mode 100644 index 000000000..689441750 --- /dev/null +++ b/packages/web/src/LoginPage.svelte @@ -0,0 +1,115 @@ + + +
+
DbGate
+
+ +
+
Log In
+ + + + +
+ { + enableApi(); + const resp = await apiCall('auth/login', e.detail); + if (resp.error) { + internalRedirectTo(`/?page=not-logged&error=${encodeURIComponent(resp.error)}`); + return; + } + const { accessToken } = resp; + if (accessToken) { + localStorage.setItem('accessToken', accessToken); + internalRedirectTo('/'); + return; + } + internalRedirectTo(`/?page=not-logged`); + }} + /> +
+
+
+
+
+ + diff --git a/packages/web/src/NotLoggedPage.svelte b/packages/web/src/NotLoggedPage.svelte new file mode 100644 index 000000000..0dad204f0 --- /dev/null +++ b/packages/web/src/NotLoggedPage.svelte @@ -0,0 +1,52 @@ + + +
+
Sorry, you are not authorized to run DbGate
+ {#if error} +
{error}
+ {/if} + +
+ + +
+
+ + diff --git a/packages/web/src/clientAuth.ts b/packages/web/src/clientAuth.ts new file mode 100644 index 000000000..8d72a697a --- /dev/null +++ b/packages/web/src/clientAuth.ts @@ -0,0 +1,107 @@ +import { apiCall, enableApi } from './utility/api'; +import { getConfig } from './utility/metadataLoaders'; + +export function isOauthCallback() { + const params = new URLSearchParams(location.search); + const sentCode = params.get('code'); + const sentState = params.get('state'); + + return ( + sentCode && sentState && sentState.startsWith('dbg-oauth:') && sentState == sessionStorage.getItem('oauthState') + ); +} + +export function handleOauthCallback() { + const params = new URLSearchParams(location.search); + const sentCode = params.get('code'); + + if (isOauthCallback()) { + sessionStorage.removeItem('oauthState'); + apiCall('auth/oauth-token', { + code: sentCode, + redirectUri: location.origin + location.pathname, + }).then(authResp => { + const { accessToken, error, errorMessage } = authResp; + + if (accessToken) { + console.log('Settings access token from OAUTH'); + localStorage.setItem('accessToken', accessToken); + internalRedirectTo('/'); + } else { + console.log('Error when processing OAUTH callback', error || errorMessage); + internalRedirectTo(`/?page=not-logged&error=${error || errorMessage}`); + } + }); + + return true; + } + + return false; +} + +export async function handleAuthOnStartup(config) { + if (config.oauth) { + console.log('OAUTH callback URL:', location.origin + location.pathname); + } + if (config.oauth || config.isLoginForm) { + if (localStorage.getItem('accessToken')) { + return; + } + + redirectToLogin(config); + } +} + +export async function redirectToLogin(config = null, force = false) { + if (!config) { + enableApi(); + config = await getConfig(); + } + + if (config.isLoginForm) { + if (!force) { + const params = new URLSearchParams(location.search); + if (params.get('page') == 'login' || params.get('page') == 'not-logged') { + return; + } + } + internalRedirectTo('/?page=login'); + return; + } + + if (config.oauth) { + const state = `dbg-oauth:${Math.random().toString().substr(2)}`; + sessionStorage.setItem('oauthState', state); + console.log('Redirecting to OAUTH provider'); + location.replace( + `${config.oauth}?client_id=dbgate&response_type=code&redirect_uri=${encodeURIComponent( + location.origin + location.pathname + )}&state=${encodeURIComponent(state)}` + ); + return; + } +} + +export function internalRedirectTo(path) { + const index = location.pathname.lastIndexOf('/'); + const newPath = index >= 0 ? location.pathname.substring(0, index) + path : path; + location.replace(newPath); +} + +export async function doLogout() { + enableApi(); + const config = await getConfig(); + if (config.oauth) { + localStorage.removeItem('accessToken'); + if (config.oauthLogout) { + window.location.href = config.oauthLogout; + } else { + internalRedirectTo('/?page=not-logged'); + } + } else if (config.isLoginForm) { + localStorage.removeItem('accessToken'); + internalRedirectTo('/?page=not-logged'); + } else { + window.location.href = 'config/logout'; + } +} diff --git a/packages/web/src/commands/stdCommands.ts b/packages/web/src/commands/stdCommands.ts index b0260cce5..6a0b7f054 100644 --- a/packages/web/src/commands/stdCommands.ts +++ b/packages/web/src/commands/stdCommands.ts @@ -36,6 +36,7 @@ import runCommand from './runCommand'; import { openWebLink } from '../utility/exportFileTools'; import { getSettings } from '../utility/metadataLoaders'; import { isMac } from '../utility/common'; +import { doLogout, internalRedirectTo } from '../clientAuth'; // function themeCommand(theme: ThemeDefinition) { // return { @@ -548,9 +549,7 @@ registerCommand({ category: 'App', name: 'Logout', testEnabled: () => getCurrentConfig()?.login != null, - onClick: () => { - window.location.href = 'config/logout'; - }, + onClick: doLogout, }); export function registerFileCommands({ diff --git a/packages/web/src/forms/TextField.svelte b/packages/web/src/forms/TextField.svelte index 55f9a3506..e373dce94 100644 --- a/packages/web/src/forms/TextField.svelte +++ b/packages/web/src/forms/TextField.svelte @@ -4,6 +4,7 @@ export let value; export let focused = false; export let domEditor = undefined; + export let autocomplete = 'new-password'; if (focused) onMount(() => domEditor.focus()); @@ -17,5 +18,5 @@ on:click bind:this={domEditor} on:keydown - autocomplete="new-password" + {autocomplete} /> diff --git a/packages/web/src/main.ts b/packages/web/src/main.ts index 2a72600c5..2bfcb4749 100644 --- a/packages/web/src/main.ts +++ b/packages/web/src/main.ts @@ -3,14 +3,41 @@ import './utility/connectionsPinger'; import './utility/changeCurrentDbByTab'; import './commands/stdCommands'; import localStorageGarbageCollector from './utility/localStorageGarbageCollector'; +import { handleOauthCallback } from './clientAuth'; +import LoginPage from './LoginPage.svelte'; +import NotLoggedPage from './NotLoggedPage.svelte'; + +const isOauthCallback = handleOauthCallback(); + +const params = new URLSearchParams(location.search); +const page = params.get('page'); localStorageGarbageCollector(); -const app = new App({ - target: document.body, - props: {}, -}); +function createApp() { + if (isOauthCallback) { + return null; + } -// const app = null; + switch (page) { + case 'login': + return new LoginPage({ + target: document.body, + props: {}, + }); + case 'not-logged': + return new NotLoggedPage({ + target: document.body, + props: {}, + }); + } + + return new App({ + target: document.body, + props: {}, + }); +} + +const app = createApp(); export default app; diff --git a/packages/web/src/utility/api.ts b/packages/web/src/utility/api.ts index c4f6694b1..e55c839c5 100644 --- a/packages/web/src/utility/api.ts +++ b/packages/web/src/utility/api.ts @@ -4,10 +4,21 @@ import { writable } from 'svelte/store'; import getElectron from './getElectron'; // import socket from './socket'; import { showSnackbarError } from '../utility/snackbar'; +import { isOauthCallback, redirectToLogin } from '../clientAuth'; let eventSource; let apiLogging = false; // let cacheCleanerRegistered; +let apiDisabled = false; +const disabledOnOauth = isOauthCallback(); + +export function disableApi() { + apiDisabled = true; +} + +export function enableApi() { + apiDisabled = false; +} function wantEventSource() { if (!eventSource) { @@ -17,9 +28,9 @@ function wantEventSource() { } function processApiResponse(route, args, resp) { - if (apiLogging) { - console.log('<<< API RESPONSE', route, args, resp); - } + // if (apiLogging) { + // console.log('<<< API RESPONSE', route, args, resp); + // } if (resp?.apiErrorMessage) { showSnackbarError('API error:' + resp?.apiErrorMessage); @@ -35,6 +46,14 @@ export async function apiCall(route: string, args: {} = undefined) { if (apiLogging) { console.log('>>> API CALL', route, args); } + if (apiDisabled) { + console.log('API disabled!!', route); + return; + } + if (disabledOnOauth && route != 'auth/oauth-token') { + console.log('API disabled because oauth callback!!', route); + return; + } const electron = getElectron(); if (electron) { @@ -51,6 +70,18 @@ export async function apiCall(route: string, args: {} = undefined) { body: JSON.stringify(args), }); + if (resp.status == 401 && !apiDisabled) { + const params = new URLSearchParams(location.search); + + disableApi(); + console.log('Disabling API', route); + if (params.get('page') != 'login' && params.get('page') != 'not-logged') { + // unauthorized + redirectToLogin(); + } + return; + } + const json = await resp.json(); return processApiResponse(route, args, json); } diff --git a/packages/web/src/utility/resolveApi.ts b/packages/web/src/utility/resolveApi.ts index e15506e6b..f2c00b555 100644 --- a/packages/web/src/utility/resolveApi.ts +++ b/packages/web/src/utility/resolveApi.ts @@ -15,5 +15,10 @@ export default function resolveApi() { export function resolveApiHeaders() { const electron = getElectron(); - return {}; + const res = {}; + const accessToken = localStorage.getItem('accessToken'); + if (accessToken) { + res['Authorization'] = `Bearer ${accessToken}`; + } + return res; } diff --git a/yarn.lock b/yarn.lock index f5a01b77e..ccb5e7f07 100644 --- a/yarn.lock +++ b/yarn.lock @@ -1701,6 +1701,11 @@ abbrev@1: resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8" integrity sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q== +abstract-logging@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/abstract-logging/-/abstract-logging-2.0.1.tgz#6b0c371df212db7129b57d2e7fcf282b8bf1c839" + integrity sha512-2BjRTZxTPvheOvGbBslFSYOUkr+SjPtOnrLP33f+VIWLzezQpZcqVg7ja3L4dBXmzzgwT+a029jRx5PCi3JuiA== + accepts@~1.3.8: version "1.3.8" resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.8.tgz#0bf0be125b67014adcb0b0921e62db7bffe16b2e" @@ -1765,6 +1770,16 @@ acorn@^8.2.4, acorn@^8.5.0: resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.8.0.tgz#88c0187620435c7f6015803f5539dae05a9dbea8" integrity sha512-QOxyigPVrpZ2GXT+PFyZTl6TtOFc5egxHIP9IlQ+RbupQuX4RkT/Bee4/kQuC02Xkzg84JcT7oLYtDIQxp+v7w== +activedirectory2@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/activedirectory2/-/activedirectory2-2.1.0.tgz#4293f72ade8ff36e9199cf5fa5cae3818bdb947a" + integrity sha512-HaccG+/mf5NpHL1mAcLzXed4+gGlO6l7mkBi8vNIo6sTJvLoJjHgvJg12F4cy5CNcRqvPS48++s5tfdSiafn4Q== + dependencies: + abstract-logging "^2.0.0" + async "^3.1.0" + ldapjs "^2.1.0" + merge-options "^2.0.0" + adler-32@~1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/adler-32/-/adler-32-1.2.0.tgz#6a3e6bf0a63900ba15652808cb15c6813d1a5f25" @@ -2058,7 +2073,7 @@ async@^2.6.2: dependencies: lodash "^4.17.14" -async@^3.2.0, async@^3.2.3: +async@^3.1.0, async@^3.2.0, async@^3.2.3: version "3.2.4" resolved "https://registry.yarnpkg.com/async/-/async-3.2.4.tgz#2d22e00f8cddeb5fde5dd33522b56d1cf569a81c" integrity sha512-iAB+JbDEGXhyIUavoDl9WP/Jj106Kz9DEn1DPgYw5ruDn0e3Wgi3sKFm55sASdGBNOQB8F59d9qQ7deqrHA8wQ== @@ -2225,6 +2240,13 @@ babel-preset-jest@^28.1.3: babel-plugin-jest-hoist "^28.1.3" babel-preset-current-node-syntax "^1.0.0" +backoff@^2.5.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/backoff/-/backoff-2.5.0.tgz#f616eda9d3e4b66b8ca7fca79f695722c5f8e26f" + integrity sha512-wC5ihrnUXmR2douXmXLCe5O3zg3GKIyvRi/hi58a/XyRxVI+3/yM0PYueQOZXPXQ9pxBislYkw+sF9b7C/RuMA== + dependencies: + precond "0.2" + balanced-match@^1.0.0: version "1.0.2" resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" @@ -5416,6 +5438,11 @@ is-plain-obj@^1.1.0: resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-1.1.0.tgz#71a50c8429dfca773c92a390a4a03b39fcd51d3e" integrity sha512-yvkRyxmFKEOQ4pNXCmJG5AEQNlXJS5LaONXo5/cLdTZdWvsZ1ioJEonLGAosKlMWE8lwUy/bJzMjcw8az73+Fg== +is-plain-obj@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-2.1.0.tgz#45e42e37fccf1f40da8e5f76ee21515840c09287" + integrity sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA== + is-plain-object@^2.0.3, is-plain-object@^2.0.4: version "2.0.4" resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677" @@ -7044,6 +7071,27 @@ kleur@^3.0.0, kleur@^3.0.3: resolved "https://registry.yarnpkg.com/kleur/-/kleur-3.0.3.tgz#a79c9ecc86ee1ce3fa6206d1216c501f147fc07e" integrity sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w== +ldap-filter@^0.3.3: + version "0.3.3" + resolved "https://registry.yarnpkg.com/ldap-filter/-/ldap-filter-0.3.3.tgz#2b14c68a2a9d4104dbdbc910a1ca85fd189e9797" + integrity sha512-/tFkx5WIn4HuO+6w9lsfxq4FN3O+fDZeO9Mek8dCD8rTUpqzRa766BOBO7BcGkn3X86m5+cBm1/2S/Shzz7gMg== + dependencies: + assert-plus "^1.0.0" + +ldapjs@^2.1.0: + version "2.3.3" + resolved "https://registry.yarnpkg.com/ldapjs/-/ldapjs-2.3.3.tgz#06c317d3cbb5ac42fbba741e1a8b130ffcf997ab" + integrity sha512-75QiiLJV/PQqtpH+HGls44dXweviFwQ6SiIK27EqzKQ5jU/7UFrl2E5nLdQ3IYRBzJ/AVFJI66u0MZ0uofKYwg== + dependencies: + abstract-logging "^2.0.0" + asn1 "^0.2.4" + assert-plus "^1.0.0" + backoff "^2.5.0" + ldap-filter "^0.3.3" + once "^1.4.0" + vasync "^2.2.0" + verror "^1.8.1" + leaflet@^1.8.0: version "1.8.0" resolved "https://registry.yarnpkg.com/leaflet/-/leaflet-1.8.0.tgz#4615db4a22a304e8e692cae9270b983b38a2055e" @@ -7398,6 +7446,13 @@ merge-descriptors@1.0.1: resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61" integrity sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w== +merge-options@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/merge-options/-/merge-options-2.0.0.tgz#36ca5038badfc3974dbde5e58ba89d3df80882c3" + integrity sha512-S7xYIeWHl2ZUKF7SDeBhGg6rfv5bKxVBdk95s/I7wVF8d+hjLSztJ/B271cnUiF6CAFduEQ5Zn3HYwAjT16DlQ== + dependencies: + is-plain-obj "^2.0.0" + merge-stream@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" @@ -8623,6 +8678,11 @@ prebuild-install@^7.0.1, prebuild-install@^7.1.0, prebuild-install@^7.1.1: tar-fs "^2.0.0" tunnel-agent "^0.6.0" +precond@0.2: + version "0.2.3" + resolved "https://registry.yarnpkg.com/precond/-/precond-0.2.3.tgz#aa9591bcaa24923f1e0f4849d240f47efc1075ac" + integrity sha512-QCYG84SgGyGzqJ/vlMsxeXd/pgL/I94ixdNFyh1PusWmTCyVfPJjZ1K1jvHtsbfnXQs2TSkEP2fR7QiMZAnKFQ== + prelude-ls@~1.1.2: version "1.1.2" resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.1.2.tgz#21932a549f5e52ffd9a827f570e04be62a97da54" @@ -10864,6 +10924,13 @@ vary@^1, vary@~1.1.2: resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg== +vasync@^2.2.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/vasync/-/vasync-2.2.1.tgz#d881379ff3685e4affa8e775cf0fd369262a201b" + integrity sha512-Hq72JaTpcTFdWiNA4Y22Amej2GH3BFmBaKPPlDZ4/oC8HNn2ISHLkFrJU4Ds8R3jcUi7oo5Y9jcMHKjES+N9wQ== + dependencies: + verror "1.10.0" + verror@1.10.0: version "1.10.0" resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400" @@ -10873,6 +10940,15 @@ verror@1.10.0: core-util-is "1.0.2" extsprintf "^1.2.0" +verror@^1.8.1: + version "1.10.1" + resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.1.tgz#4bf09eeccf4563b109ed4b3d458380c972b0cdeb" + integrity sha512-veufcmxri4e3XSrT0xwfUR7kguIkaxBeosDg00yDWhk49wdwkSUrvvsm7nc75e1PUyvIeZj6nS8VQRYz2/S4Xg== + dependencies: + assert-plus "^1.0.0" + core-util-is "1.0.2" + extsprintf "^1.2.0" + vm-browserify@^1.0.1: version "1.1.2" resolved "https://registry.yarnpkg.com/vm-browserify/-/vm-browserify-1.1.2.tgz#78641c488b8e6ca91a75f511e7a3b32a86e5dda0"