Skip to content
Snippets Groups Projects
Unverified Commit 35ba4b79 authored by Marius Friess's avatar Marius Friess Committed by GitHub
Browse files

Add unit tests for BrowserModule (#56)

* Add Browser tests

* fix lint error
parent 7fdb1c42
No related branches found
No related tags found
No related merge requests found
......@@ -5,5 +5,6 @@
"testRegex": ".(e2e-)?spec.ts$",
"transform": {
"^.+\\.(t|j)s$": "ts-jest"
}
},
"collectCoverageFrom": ["src/**/*.ts", "test/**/*.ts"]
}
import { Test, TestingModule } from '@nestjs/testing';
import { BrowserService } from './browser.service';
import { Server } from 'socket.io';
import { Channel } from '../channel/channel';
describe('BrowserService', () => {
let service: BrowserService;
beforeEach(async () => {
const module: TestingModule = await Test.createTestingModule({
imports: [],
providers: [BrowserService],
}).compile();
service = module.get<BrowserService>(BrowserService);
});
afterEach(async () => {
await service.close();
});
/**
* Single test case: Verifies if the BrowserService is defined.
*/
it('should be defined', () => {
expect(service).toBeDefined();
});
describe('openWebsite', () => {
it('should create a browser context for a new channelId', async () => {
const channelId = 'test-channel-id';
const url = 'https://example.com';
const server = {
to: () => {
return {
emit: jest.fn(),
};
},
} as unknown as Server;
const peerId = await service.openWebsite(channelId, url, server);
expect(peerId).toBeDefined();
expect(service.getPeerId(channelId)).toEqual(peerId);
});
});
describe('browserInteraction', () => {
it('should execute the browser interaction functions', async () => {
const channelId = 'test-channel-id';
const url = 'https://example.com';
const server = {
to: () => {
return {
emit: jest.fn(),
};
},
} as unknown as Server;
const peerId = await service.openWebsite(channelId, url, server);
expect(peerId).toBeDefined();
await service.moveMouse(channelId, 0, 0);
await service.mouseDown(channelId);
await service.mouseUp(channelId);
await service.keyDown(channelId, 'a');
await service.keyUp(channelId, 'a');
await service.scroll(channelId, 0);
await service.navigateForward(channelId);
await service.closeBrowserContext(channelId);
expect(service.getPeerId(channelId)).toBeUndefined();
});
});
describe('getFromChannel', () => {
it('should return the browser context associated with a channel', async () => {
const channelId = 'test-channel-id';
const url = 'https://example.com';
const server = {
to: () => {
return {
emit: jest.fn(),
};
},
} as unknown as Server;
const peerId = await service.openWebsite(channelId, url, server);
expect(peerId).toBeDefined();
const channel = {
id: channelId,
} as Channel;
const browser = service.getFromChannel(channel);
expect(browser).toBeDefined();
});
});
describe('closeBrowserContext', () => {
it('should close the browser context associated with a channel', async () => {
const channelId = 'test-channel-id';
const url = 'https://example.com';
const server = {
to: () => {
return {
emit: jest.fn(),
};
},
} as unknown as Server;
const peerId = await service.openWebsite(channelId, url, server);
expect(peerId).toBeDefined();
const channel = {
id: channelId,
} as Channel;
const browser = service.getFromChannel(channel);
expect(browser).toBeDefined();
await service.closeBrowserContext(channelId);
const browser2 = service.getFromChannel(channel);
expect(browser2).toBeNull();
});
});
});
......@@ -21,6 +21,8 @@ export class Browser {
*
* @param browser - The Puppeteer browser instance.
* @param url - The initial URL to open.
* @param server - The Socket.IO server instance.
* @param channelId - The ID of the channel.
*/
constructor(
private browser: PuppeteerBrowser,
......@@ -38,7 +40,7 @@ export class Browser {
LOGGER.debug(`Opening page with url: ${this.url}`);
this.page = await this.browser.newPage();
await installMouseHelper(this.page);
await this.installMouseHelper(this.page);
await this.page.goto(this.url);
await this.page.setViewport({ width: 1920, height: 1080 });
......@@ -65,6 +67,7 @@ export class Browser {
await this.page.bringToFront();
/* istanbul ignore next */
const id = await extensionPage.evaluate(async () => {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
......@@ -167,25 +170,25 @@ export class Browser {
await this.browser.close();
}
}
}
/**
* Helper function to install mouse tracking for puppeteer pages.
*
* @param page - The puppeteer page to install the mouse tracking on.
*/
async function installMouseHelper(page) {
await page.evaluateOnNewDocument(() => {
// Install mouse helper only for top-level frame.
if (window !== window.parent) {
return;
}
window.addEventListener(
'DOMContentLoaded',
() => {
const box = document.createElement('puppeteer-mouse-pointer');
const styleElement = document.createElement('style');
styleElement.innerHTML = `
/**
* Helper function to install mouse tracking for puppeteer pages.
*
* @param page - The puppeteer page to install the mouse tracking on.
*/
private async installMouseHelper(page) {
await page.evaluateOnNewDocument(() => {
// Install mouse helper only for top-level frame.
if (window !== window.parent) {
return;
}
// istanbul ignore next
window.addEventListener(
'DOMContentLoaded',
() => {
const box = document.createElement('puppeteer-mouse-pointer');
const styleElement = document.createElement('style');
styleElement.innerHTML = `
puppeteer-mouse-pointer {
pointer-events: none;
position: absolute;
......@@ -222,43 +225,44 @@ async function installMouseHelper(page) {
border-color: rgba(0,255,0,0.9);
}
`;
document.head.appendChild(styleElement);
document.body.appendChild(box);
document.addEventListener(
'mousemove',
(event) => {
box.style.left = event.pageX + 'px';
box.style.top = event.pageY + 'px';
updateButtons(event.buttons);
},
true,
);
document.addEventListener(
'mousedown',
(event) => {
updateButtons(event.buttons);
box.classList.add('button-' + event.which);
},
true,
);
document.addEventListener(
'mouseup',
(event) => {
updateButtons(event.buttons);
box.classList.remove('button-' + event.which);
},
true,
);
document.head.appendChild(styleElement);
document.body.appendChild(box);
document.addEventListener(
'mousemove',
(event) => {
box.style.left = event.pageX + 'px';
box.style.top = event.pageY + 'px';
updateButtons(event.buttons);
},
true,
);
document.addEventListener(
'mousedown',
(event) => {
updateButtons(event.buttons);
box.classList.add('button-' + event.which);
},
true,
);
document.addEventListener(
'mouseup',
(event) => {
updateButtons(event.buttons);
box.classList.remove('button-' + event.which);
},
true,
);
function updateButtons(buttons) {
for (let i = 0; i < 5; i++) {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
box.classList.toggle('button-' + i, buttons & (1 << i));
function updateButtons(buttons) {
for (let i = 0; i < 5; i++) {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
box.classList.toggle('button-' + i, buttons & (1 << i));
}
}
}
},
false,
);
});
},
false,
);
});
}
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment