mirror of
https://github.com/ajayyy/SponsorBlock.git
synced 2024-11-10 09:07:45 +01:00
230 lines
No EOL
9.6 KiB
TypeScript
230 lines
No EOL
9.6 KiB
TypeScript
import { Builder, By, until, WebDriver, WebElement } from "selenium-webdriver";
|
|
import * as Chrome from "selenium-webdriver/chrome";
|
|
import * as Path from "path";
|
|
|
|
import * as fs from "fs";
|
|
|
|
test("Selenium Chrome test", async () => {
|
|
let driver: WebDriver;
|
|
try {
|
|
driver = await setup();
|
|
} catch (e) {
|
|
console.warn("A browser is probably not installed, skipping selenium tests");
|
|
console.warn(e);
|
|
|
|
return;
|
|
}
|
|
|
|
try {
|
|
await waitForInstall(driver);
|
|
// This video has no ads
|
|
await goToVideo(driver, "jNQXAC9IVRw");
|
|
|
|
await createSegment(driver, "4", "10.33", "0:04.000 to 0:10.330");
|
|
|
|
await editSegments(driver, 0, "0:04.000", "0:10.330", "5", "13.211", "0:05.000 to 0:13.211", false);
|
|
await autoskipSegment(driver, 5, 13.211);
|
|
|
|
await setSegmentCategory(driver, 0, 1, false);
|
|
await setSegmentActionType(driver, 0, 1, false);
|
|
await editSegments(driver, 0, "0:05.000", "0:13.211", "5", "7.5", "0:05.000 to 0:07.500", false);
|
|
await muteSkipSegment(driver, 5, 7.5);
|
|
|
|
// Full video
|
|
await setSegmentActionType(driver, 0, 2, false);
|
|
await driver.wait(until.elementIsNotVisible(await getDisplayTimeBox(driver, 0)));
|
|
|
|
await toggleWhitelist(driver);
|
|
await toggleWhitelist(driver);
|
|
|
|
} catch (e) {
|
|
// Save file incase there is a layout change
|
|
const source = await driver.getPageSource();
|
|
|
|
if (!fs.existsSync("./test-results")) fs.mkdirSync("./test-results");
|
|
fs.writeFileSync("./test-results/source.html", source);
|
|
|
|
throw e;
|
|
} finally {
|
|
await driver.quit();
|
|
}
|
|
}, 100_000);
|
|
|
|
async function setup(): Promise<WebDriver> {
|
|
const options = new Chrome.Options();
|
|
options.addArguments("--load-extension=" + Path.join(__dirname, "../dist/"));
|
|
options.addArguments("--mute-audio");
|
|
options.addArguments("--disable-features=PreloadMediaEngagementData, MediaEngagementBypassAutoplayPolicies");
|
|
options.addArguments("--headless=new");
|
|
options.addArguments("--window-size=1920,1080");
|
|
|
|
const driver = await new Builder().forBrowser("chrome").setChromeOptions(options).build();
|
|
driver.manage().setTimeouts({
|
|
implicit: 5000
|
|
});
|
|
|
|
return driver;
|
|
}
|
|
|
|
async function waitForInstall(driver: WebDriver, startingTab = 0): Promise<void> {
|
|
// Selenium only knows about the one tab it's on,
|
|
// so we can't wait for the help page to appear
|
|
await driver.sleep(3000);
|
|
|
|
const handles = await driver.getAllWindowHandles();
|
|
await driver.switchTo().window(handles[startingTab]);
|
|
}
|
|
|
|
async function goToVideo(driver: WebDriver, videoId: string): Promise<void> {
|
|
await driver.get("https://www.youtube.com/watch?v=" + videoId);
|
|
await driver.wait(until.elementIsVisible(await driver.findElement(By.css(".ytd-video-primary-info-renderer, #above-the-fold"))));
|
|
}
|
|
|
|
async function createSegment(driver: WebDriver, startTime: string, endTime: string, expectedDisplayedTime: string): Promise<void> {
|
|
const startSegmentButton = await driver.findElement(By.id("startSegmentButton"));
|
|
const cancelSegmentButton = await driver.findElement(By.id("cancelSegmentButton"));
|
|
await driver.executeScript("document.querySelector('video').currentTime = " + startTime);
|
|
|
|
await startSegmentButton.click();
|
|
await driver.wait(until.elementIsVisible(cancelSegmentButton));
|
|
|
|
await driver.executeScript("document.querySelector('video').currentTime = " + endTime);
|
|
|
|
await startSegmentButton.click();
|
|
await driver.wait(until.elementIsNotVisible(cancelSegmentButton));
|
|
|
|
const submitButton = await driver.findElement(By.id("submitButton"));
|
|
await submitButton.click();
|
|
|
|
const sponsorTimeDisplays = await driver.findElements(By.className("sponsorTimeDisplay"));
|
|
const sponsorTimeDisplay = sponsorTimeDisplays[sponsorTimeDisplays.length - 1];
|
|
await driver.wait(until.elementTextIs(sponsorTimeDisplay, expectedDisplayedTime));
|
|
}
|
|
|
|
async function editSegments(driver: WebDriver, index: number, expectedStartTimeBox: string, expectedEndTimeBox: string,
|
|
startTime: string, endTime: string, expectedDisplayedTime: string, openSubmitBox: boolean): Promise<void> {
|
|
|
|
if (openSubmitBox) {
|
|
const submitButton = await driver.findElement(By.id("submitButton"));
|
|
await submitButton.click();
|
|
}
|
|
|
|
let editButton = await driver.findElement(By.id("sponsorTimeEditButtonSubmissionNotice" + index));
|
|
const sponsorTimeDisplay = await getDisplayTimeBox(driver, index);
|
|
await sponsorTimeDisplay.click();
|
|
// Ensure edit time appears
|
|
await driver.findElement(By.id("submittingTime0SubmissionNotice" + index));
|
|
|
|
// Try the edit button too
|
|
await editButton.click();
|
|
await editButton.click();
|
|
|
|
const startTimeBox = await getStartTimeBox(driver, index, expectedStartTimeBox);
|
|
await startTimeBox.clear();
|
|
await startTimeBox.sendKeys(startTime);
|
|
|
|
const endTimeBox = await getEndTimeBox(driver, index, expectedEndTimeBox);
|
|
await endTimeBox.clear();
|
|
await endTimeBox.sendKeys(endTime);
|
|
|
|
editButton = await driver.findElement(By.id("sponsorTimeEditButtonSubmissionNotice" + index));
|
|
await editButton.click();
|
|
|
|
await getDisplayTimeBox(driver, index, expectedDisplayedTime);
|
|
}
|
|
|
|
async function getStartTimeBox(driver: WebDriver, index: number, expectedStartTimeBox: string): Promise<WebElement> {
|
|
const startTimeBox = await driver.findElement(By.id("submittingTime0SubmissionNotice" + index));
|
|
expect((await startTimeBox.getAttribute("value"))).toBe(expectedStartTimeBox);
|
|
return startTimeBox;
|
|
}
|
|
|
|
async function getEndTimeBox(driver: WebDriver, index: number, expectedEndTimeBox: string): Promise<WebElement> {
|
|
const endTimeBox = await driver.findElement(By.id("submittingTime1SubmissionNotice" + index));
|
|
expect((await endTimeBox.getAttribute("value"))).toBe(expectedEndTimeBox);
|
|
return endTimeBox;
|
|
}
|
|
|
|
async function getDisplayTimeBox(driver: WebDriver, index: number, expectedDisplayedTime?: string): Promise<WebElement> {
|
|
const sponsorTimeDisplay = (await driver.findElements(By.className("sponsorTimeDisplay")))[index];
|
|
if (expectedDisplayedTime) {
|
|
driver.wait(until.elementTextIs(sponsorTimeDisplay, expectedDisplayedTime));
|
|
}
|
|
|
|
return sponsorTimeDisplay;
|
|
}
|
|
|
|
async function setSegmentCategory(driver: WebDriver, index: number, categoryIndex: number, openSubmitBox: boolean): Promise<void> {
|
|
if (openSubmitBox) {
|
|
const submitButton = await driver.findElement(By.id("submitButton"));
|
|
await submitButton.click();
|
|
}
|
|
|
|
const categorySelection = await driver.findElement(By.css(`#sponsorTimeCategoriesSubmissionNotice${index} > option:nth-child(${categoryIndex + 1})`));
|
|
await categorySelection.click();
|
|
}
|
|
|
|
async function setSegmentActionType(driver: WebDriver, index: number, actionTypeIndex: number, openSubmitBox: boolean): Promise<void> {
|
|
if (openSubmitBox) {
|
|
const submitButton = await driver.findElement(By.id("submitButton"));
|
|
await submitButton.click();
|
|
}
|
|
|
|
const actionTypeSelection = await driver.findElement(By.css(`#sponsorTimeActionTypesSubmissionNotice${index} > option:nth-child(${actionTypeIndex + 1})`));
|
|
await actionTypeSelection.click();
|
|
}
|
|
|
|
async function autoskipSegment(driver: WebDriver, startTime: number, endTime: number): Promise<void> {
|
|
const video = await driver.findElement(By.css("video"));
|
|
|
|
await driver.executeScript("document.querySelector('video').currentTime = " + (startTime - 0.5));
|
|
await driver.executeScript("document.querySelector('video').play()");
|
|
|
|
await driver.sleep(1300);
|
|
expect(parseFloat(await video.getAttribute("currentTime"))).toBeGreaterThan(endTime);
|
|
await driver.executeScript("document.querySelector('video').pause()");
|
|
}
|
|
|
|
async function muteSkipSegment(driver: WebDriver, startTime: number, endTime: number): Promise<void> {
|
|
const duration = endTime - startTime;
|
|
const video = await driver.findElement(By.css("video"));
|
|
|
|
await driver.executeScript("document.querySelector('video').currentTime = " + (startTime - 0.5));
|
|
await driver.executeScript("document.querySelector('video').play()");
|
|
|
|
await driver.sleep(1300);
|
|
expect(await video.getAttribute("muted")).toEqual("true");
|
|
|
|
await driver.sleep(duration * 1000 + 300);
|
|
expect(await video.getAttribute("muted")).toBeNull(); // Default is null for some reason
|
|
await driver.executeScript("document.querySelector('video').pause()");
|
|
}
|
|
|
|
async function toggleWhitelist(driver: WebDriver): Promise<void> {
|
|
const popupButton = await driver.findElement(By.id("infoButton"));
|
|
const rightControls = await driver.findElement(By.css(".ytp-right-controls"));
|
|
await driver.actions().move({ origin: rightControls }).perform();
|
|
if ((await popupButton.getCssValue("display")) !== "none") {
|
|
await driver.actions().move({ origin: popupButton }).perform();
|
|
await popupButton.click();
|
|
}
|
|
|
|
const popupFrame = await driver.findElement(By.css("#sponsorBlockPopupContainer iframe"));
|
|
await driver.switchTo().frame(popupFrame);
|
|
|
|
const whitelistButton = await driver.findElement(By.id("whitelistButton"));
|
|
await driver.wait(until.elementIsVisible(whitelistButton));
|
|
|
|
const whitelistText = await driver.findElement(By.id("whitelistChannel"));
|
|
const whitelistDisplayed = await whitelistText.isDisplayed();
|
|
|
|
await whitelistButton.click();
|
|
if (whitelistDisplayed) {
|
|
const unwhitelistText = await driver.findElement(By.id("unwhitelistChannel"));
|
|
await driver.wait(until.elementIsVisible(unwhitelistText));
|
|
} else {
|
|
await driver.wait(until.elementIsVisible(whitelistText));
|
|
}
|
|
|
|
await driver.switchTo().defaultContent();
|
|
} |