mirror of
https://github.com/cypress-io/cypress.git
synced 2026-02-13 18:50:18 -06:00
225 lines
5.4 KiB
JavaScript
225 lines
5.4 KiB
JavaScript
const _ = require('lodash')
|
|
const debug = require('debug')('cypress:server:auth')
|
|
const express = require('express')
|
|
const os = require('os')
|
|
const pkg = require('@packages/root')
|
|
const Promise = require('bluebird')
|
|
const { shell } = require('electron')
|
|
const url = require('url')
|
|
|
|
const errors = require('../errors')
|
|
const machineId = require('../util/machine_id')
|
|
const random = require('../util/random')
|
|
const user = require('../user')
|
|
const windows = require('./windows')
|
|
|
|
let app
|
|
let authCallback
|
|
let authState
|
|
let openExternalAttempted = false
|
|
let authRedirectReached = false
|
|
let server
|
|
|
|
const _buildLoginRedirectUrl = (server) => {
|
|
const { port } = server.address()
|
|
|
|
return `http://127.0.0.1:${port}/redirect-to-auth`
|
|
}
|
|
|
|
const _buildFullLoginUrl = (baseLoginUrl, server, utmCode) => {
|
|
const { port } = server.address()
|
|
|
|
if (!authState) {
|
|
authState = random.id(32)
|
|
}
|
|
|
|
const authUrl = url.parse(baseLoginUrl)
|
|
|
|
return machineId.machineId()
|
|
.then((id) => {
|
|
authUrl.query = {
|
|
port,
|
|
state: authState,
|
|
machineId: id,
|
|
cypressVersion: pkg.version,
|
|
platform: os.platform(),
|
|
}
|
|
|
|
if (utmCode) {
|
|
authUrl.query = {
|
|
utm_source: 'Test Runner',
|
|
utm_medium: utmCode,
|
|
utm_campaign: 'Log In',
|
|
...authUrl.query,
|
|
}
|
|
}
|
|
|
|
return authUrl.format()
|
|
})
|
|
}
|
|
|
|
const _getOriginFromUrl = (originalUrl) => {
|
|
const parsedUrl = url.parse(originalUrl)
|
|
|
|
return url.format(_.pick(parsedUrl, ['protocol', 'slashes', 'hostname', 'port']))
|
|
}
|
|
|
|
/**
|
|
* @returns a promise that is resolved with a user when auth is complete or rejected when it fails
|
|
*/
|
|
const start = (onMessage, utmCode) => {
|
|
function sendMessage (type, name, arg1) {
|
|
onMessage({
|
|
type,
|
|
name,
|
|
message: errors.getMsgByType(name, arg1),
|
|
browserOpened: authRedirectReached,
|
|
})
|
|
}
|
|
|
|
authRedirectReached = false
|
|
|
|
return user.getBaseLoginUrl()
|
|
.then((baseLoginUrl) => {
|
|
return _launchServer(baseLoginUrl, sendMessage, utmCode)
|
|
})
|
|
.then(() => {
|
|
return _buildLoginRedirectUrl(server)
|
|
})
|
|
.then((loginRedirectUrl) => {
|
|
debug('Trying to open native auth to URL ', loginRedirectUrl)
|
|
|
|
return _launchNativeAuth(loginRedirectUrl, sendMessage)
|
|
.then(() => {
|
|
debug('openExternal completed')
|
|
})
|
|
})
|
|
.then(() => {
|
|
return Promise.fromCallback((cb) => {
|
|
authCallback = cb
|
|
})
|
|
})
|
|
.finally(() => {
|
|
_stopServer()
|
|
windows.focusMainWindow()
|
|
})
|
|
}
|
|
|
|
/**
|
|
* @returns the currently running auth server instance, launches one if there is not one
|
|
*/
|
|
const _launchServer = (baseLoginUrl, sendMessage, utmCode) => {
|
|
if (!server) {
|
|
// launch an express server to listen for the auth callback from dashboard
|
|
const origin = _getOriginFromUrl(baseLoginUrl)
|
|
|
|
debug('Launching auth server with origin', origin)
|
|
app = express()
|
|
|
|
app.get('/redirect-to-auth', (req, res) => {
|
|
authRedirectReached = true
|
|
|
|
_buildFullLoginUrl(baseLoginUrl, server, utmCode)
|
|
.then((fullLoginUrl) => {
|
|
debug('Received GET to /redirect-to-auth, redirecting: %o', { fullLoginUrl })
|
|
|
|
res.redirect(303, fullLoginUrl)
|
|
|
|
sendMessage('info', 'AUTH_BROWSER_LAUNCHED')
|
|
})
|
|
})
|
|
|
|
app.get('/auth', (req, res) => {
|
|
debug('Received GET to /auth with query params %o', req.query)
|
|
|
|
const redirectToStatus = (status) => {
|
|
res.redirect(`${baseLoginUrl}?status=${status}`)
|
|
}
|
|
|
|
/**
|
|
* Cypress Dashboard can redirect to us with an error; or, if Electron's shell.openExternal
|
|
* is bugging out, `authCallback` can be undefined and reaching this point makes no sense.
|
|
* @see https://github.com/cypress-io/cypress/pull/5243
|
|
*/
|
|
if (_.get(req.query, 'status') === 'error' || !authCallback) {
|
|
if (authCallback) {
|
|
authCallback(new Error('There was an error authenticating to the Cypress dashboard.'))
|
|
}
|
|
|
|
return redirectToStatus('error')
|
|
}
|
|
|
|
const { state, name, email, access_token } = req.query
|
|
|
|
if (state === authState && access_token) {
|
|
const userObj = {
|
|
name,
|
|
email,
|
|
authToken: access_token,
|
|
}
|
|
|
|
return user.set(userObj)
|
|
.then(() => {
|
|
authCallback(undefined, userObj)
|
|
redirectToStatus('success')
|
|
})
|
|
.catch((err) => {
|
|
authCallback(err)
|
|
redirectToStatus('error')
|
|
})
|
|
}
|
|
|
|
redirectToStatus('error')
|
|
})
|
|
|
|
return new Promise.fromCallback((cb) => {
|
|
server = app.listen(0, '127.0.0.1', cb)
|
|
})
|
|
}
|
|
|
|
return Promise.resolve()
|
|
}
|
|
|
|
const _stopServer = () => {
|
|
debug('Closing auth server')
|
|
if (server) {
|
|
server.close()
|
|
server = undefined
|
|
}
|
|
|
|
app = undefined
|
|
authState = undefined
|
|
authCallback = undefined
|
|
openExternalAttempted = false
|
|
authRedirectReached = false
|
|
}
|
|
|
|
const _launchNativeAuth = Promise.method((loginUrl, sendMessage) => {
|
|
const warnCouldNotLaunch = () => {
|
|
if (openExternalAttempted && !authRedirectReached) {
|
|
sendMessage('warning', 'AUTH_COULD_NOT_LAUNCH_BROWSER', loginUrl)
|
|
}
|
|
}
|
|
|
|
warnCouldNotLaunch()
|
|
|
|
setTimeout(warnCouldNotLaunch, 4000)
|
|
|
|
openExternalAttempted = true
|
|
|
|
return shell.openExternal(loginUrl)
|
|
.catch((err) => {
|
|
debug('Error launching native auth: %o', { err })
|
|
warnCouldNotLaunch()
|
|
})
|
|
})
|
|
|
|
module.exports = {
|
|
_buildFullLoginUrl,
|
|
_getOriginFromUrl,
|
|
_launchServer,
|
|
_launchNativeAuth,
|
|
_stopServer,
|
|
start,
|
|
}
|