feat: add e2e test for logout

This commit is contained in:
Joe Previte 2021-03-17 16:23:17 -07:00
parent 5cec6208d0
commit 090687d057
No known key found for this signature in database
GPG Key ID: 2C91590C6B742C24
4 changed files with 68 additions and 8 deletions

View File

@ -1,3 +1,3 @@
export enum Cookie { export enum Cookie {
Key = "key", Key = 'key',
} }

View File

@ -9,8 +9,7 @@ import { registerThemingParticipant, IThemeService } from 'vs/platform/theme/com
import { MenuBarVisibility, getTitleBarStyle, IWindowOpenable, getMenuBarVisibility } from 'vs/platform/windows/common/windows'; import { MenuBarVisibility, getTitleBarStyle, IWindowOpenable, getMenuBarVisibility } from 'vs/platform/windows/common/windows';
import { IContextKeyService } from 'vs/platform/contextkey/common/contextkey'; import { IContextKeyService } from 'vs/platform/contextkey/common/contextkey';
import { IAction, Action, SubmenuAction, Separator } from 'vs/base/common/actions'; import { IAction, Action, SubmenuAction, Separator } from 'vs/base/common/actions';
import * as DOM from 'vs/base/browser/dom'; import { addDisposableListener, Dimension, EventType, getCookieValue } from 'vs/base/browser/dom';
import { addDisposableListener, Dimension, EventType } from 'vs/base/browser/dom';
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding'; import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
import { isMacintosh, isWeb, isIOS, isNative } from 'vs/base/common/platform'; import { isMacintosh, isWeb, isIOS, isNative } from 'vs/base/common/platform';
import { IConfigurationService, IConfigurationChangeEvent } from 'vs/platform/configuration/common/configuration'; import { IConfigurationService, IConfigurationChangeEvent } from 'vs/platform/configuration/common/configuration';
@ -717,8 +716,8 @@ export class CustomMenubarControl extends MenubarControl {
webNavigationActions.push(new Action('logout', localize('logout', "Log out"), undefined, true, webNavigationActions.push(new Action('logout', localize('logout', "Log out"), undefined, true,
async (event?: MouseEvent) => { async (event?: MouseEvent) => {
const COOKIE_KEY = 'key'; const COOKIE_KEY = Cookie.Key;
const loginCookie = DOM.getCookieValue(COOKIE_KEY); const loginCookie = getCookieValue(COOKIE_KEY);
this.logService.info('Logging out of code-server'); this.logService.info('Logging out of code-server');
@ -735,7 +734,7 @@ export class CustomMenubarControl extends MenubarControl {
} else { } else {
this.logService.warn('Could not log out because we could not find cookie'); this.logService.warn('Could not log out because we could not find cookie');
} }
})) }));
return webNavigationActions; return webNavigationActions;
} }

View File

@ -3,11 +3,14 @@ import { promises as fs } from "fs"
import { RateLimiter as Limiter } from "limiter" import { RateLimiter as Limiter } from "limiter"
import * as path from "path" import * as path from "path"
import safeCompare from "safe-compare" import safeCompare from "safe-compare"
import { Cookie } from "../../../lib/vscode/src/vs/server/common/cookie"
import { rootPath } from "../constants" import { rootPath } from "../constants"
import { authenticated, getCookieDomain, redirect, replaceTemplates } from "../http" import { authenticated, getCookieDomain, redirect, replaceTemplates } from "../http"
import { hash, humanPath } from "../util" import { hash, humanPath } from "../util"
export enum Cookie {
Key = "key",
}
// RateLimiter wraps around the limiter library for logins. // RateLimiter wraps around the limiter library for logins.
// It allows 2 logins every minute and 12 logins every hour. // It allows 2 logins every minute and 12 logins every hour.
class RateLimiter { class RateLimiter {

58
test/e2e/logout.test.ts Normal file
View File

@ -0,0 +1,58 @@
import { chromium, Page, Browser, BrowserContext } from "playwright"
import { CODE_SERVER_ADDRESS, PASSWORD, E2E_VIDEO_DIR } from "../utils/constants"
describe("logout", () => {
let browser: Browser
let page: Page
let context: BrowserContext
beforeAll(async () => {
browser = await chromium.launch()
context = await browser.newContext({
recordVideo: { dir: E2E_VIDEO_DIR },
})
})
afterAll(async () => {
await browser.close()
})
beforeEach(async () => {
page = await context.newPage()
})
afterEach(async () => {
await page.close()
// Remove password from local storage
await context.clearCookies()
})
it("should be able login and logout", async () => {
await page.goto(CODE_SERVER_ADDRESS)
// Type in password
await page.fill(".password", PASSWORD)
// Click the submit button and login
await page.click(".submit")
// See the editor
const codeServerEditor = await page.isVisible(".monaco-workbench")
expect(codeServerEditor).toBeTruthy()
// Click the Application menu
await page.click("[aria-label='Application Menu']")
// See the Log out button
const logoutButton = "a.action-menu-item span[aria-label='Log out']"
expect(await page.isVisible(logoutButton))
await page.hover(logoutButton)
await page.click(logoutButton)
// it takes a second to navigate
// and since page.url comes back immediately
// we need this waitForNavigation, otherwise it will check
// before navigation has finished and fail
await page.waitForNavigation({ url: `${CODE_SERVER_ADDRESS}/login` })
const currentUrl = page.url()
expect(currentUrl).toBe(`${CODE_SERVER_ADDRESS}/login`)
})
})