From 6d210220be0875d63461829d83c6e3a59d05cf7a Mon Sep 17 00:00:00 2001 From: Chocobozzz Date: Fri, 3 Sep 2021 10:27:04 +0200 Subject: Fix NSFW filter and add tests --- client/e2e/src/po/admin-config.po.ts | 29 +++ client/e2e/src/po/login.po.ts | 20 +- client/e2e/src/po/my-account.ts | 18 ++ client/e2e/src/po/player.po.ts | 5 +- client/e2e/src/po/video-list.po.ts | 128 +++++++++++ client/e2e/src/po/video-search.po.ts | 11 + client/e2e/src/po/video-upload.po.ts | 5 + client/e2e/src/po/video-watch.po.ts | 67 +----- client/e2e/src/suites-all/videos.e2e-spec.ts | 234 ++++++++++++++++++++ .../e2e/src/suites-local/videos-list.e2e-spec.ts | 195 +++++++++++++++++ client/e2e/src/types/common.ts | 1 + client/e2e/src/urls.ts | 10 - client/e2e/src/utils.ts | 37 ---- client/e2e/src/utils/common.ts | 47 ++++ client/e2e/src/utils/elements.ts | 7 + client/e2e/src/utils/index.ts | 3 + client/e2e/src/utils/urls.ts | 10 + client/e2e/src/videos.e2e-spec.ts | 237 --------------------- 18 files changed, 716 insertions(+), 348 deletions(-) create mode 100644 client/e2e/src/po/admin-config.po.ts create mode 100644 client/e2e/src/po/video-list.po.ts create mode 100644 client/e2e/src/po/video-search.po.ts create mode 100644 client/e2e/src/suites-all/videos.e2e-spec.ts create mode 100644 client/e2e/src/suites-local/videos-list.e2e-spec.ts create mode 100644 client/e2e/src/types/common.ts delete mode 100644 client/e2e/src/urls.ts delete mode 100644 client/e2e/src/utils.ts create mode 100644 client/e2e/src/utils/common.ts create mode 100644 client/e2e/src/utils/elements.ts create mode 100644 client/e2e/src/utils/index.ts create mode 100644 client/e2e/src/utils/urls.ts delete mode 100644 client/e2e/src/videos.e2e-spec.ts (limited to 'client/e2e/src') diff --git a/client/e2e/src/po/admin-config.po.ts b/client/e2e/src/po/admin-config.po.ts new file mode 100644 index 000000000..a15184781 --- /dev/null +++ b/client/e2e/src/po/admin-config.po.ts @@ -0,0 +1,29 @@ +import { browserSleep, go } from '../utils' + +export class AdminConfigPage { + + async navigateTo (tab: 'instance-homepage' | 'basic-configuration' | 'instance-information') { + const waitTitles = { + 'instance-homepage': 'INSTANCE HOMEPAGE', + 'basic-configuration': 'APPEARANCE', + 'instance-information': 'INSTANCE' + } + + await go('/admin/config/edit-custom#' + tab) + + await $('.inner-form-title=' + waitTitles[tab]).waitForDisplayed() + } + + updateNSFWSetting (newValue: 'do_not_list' | 'blur' | 'display') { + return $('#instanceDefaultNSFWPolicy').selectByAttribute('value', newValue) + } + + updateHomepage (newValue: string) { + return $('#instanceCustomHomepageContent').setValue(newValue) + } + + async save () { + await $('input[type=submit]').click() + await browserSleep(200) + } +} diff --git a/client/e2e/src/po/login.po.ts b/client/e2e/src/po/login.po.ts index 8e3030e43..486b9a6d8 100644 --- a/client/e2e/src/po/login.po.ts +++ b/client/e2e/src/po/login.po.ts @@ -1,6 +1,7 @@ import { go } from '../utils' export class LoginPage { + async loginAsRootUser () { await go('/login') @@ -8,7 +9,7 @@ export class LoginPage { await browser.execute(`window.localStorage.setItem('no_welcome_modal', 'true')`) await $('input#username').setValue('root') - await $('input#password').setValue('test1') + await $('input#password').setValue('test' + this.getSuffix()) await browser.pause(1000) @@ -19,7 +20,24 @@ export class LoginPage { await expect(this.getLoggedInInfoElem()).toHaveText('root') } + async logout () { + await $('.logged-in-more').click() + + const logout = () => $('.dropdown-item*=Log out') + + await logout().waitForDisplayed() + await logout().click() + + await $('.login-buttons-block').waitForDisplayed() + } + private getLoggedInInfoElem () { return $('.logged-in-display-name') } + + private getSuffix () { + return browser.config.baseUrl + ? browser.config.baseUrl.slice(-1) + : '1' + } } diff --git a/client/e2e/src/po/my-account.ts b/client/e2e/src/po/my-account.ts index 85dc02805..8b5e79b5e 100644 --- a/client/e2e/src/po/my-account.ts +++ b/client/e2e/src/po/my-account.ts @@ -14,6 +14,24 @@ export class MyAccountPage { return $('a[href="/my-library/history/videos"]').click() } + // Settings + + navigateToMySettings () { + return $('a[href="/my-account"]').click() + } + + async updateNSFW (newValue: 'do_not_list' | 'blur' | 'display') { + const nsfw = $('#nsfwPolicy') + + await nsfw.waitForDisplayed() + await nsfw.scrollIntoView(false) // Avoid issues with fixed header on firefox + await nsfw.selectByAttribute('value', newValue) + + const submit = $('my-user-video-settings input[type=submit]') + await submit.scrollIntoView(false) + await submit.click() + } + // My account Videos async removeVideo (name: string) { diff --git a/client/e2e/src/po/player.po.ts b/client/e2e/src/po/player.po.ts index 372e8ab20..fca3bcdba 100644 --- a/client/e2e/src/po/player.po.ts +++ b/client/e2e/src/po/player.po.ts @@ -15,6 +15,9 @@ export class PlayerPage { waitUntilPlaylistInfo (text: string, maxTime: number) { return browser.waitUntil(async () => { + // Without this we have issues on iphone + await $('.video-js').click() + return (await $('.video-js .vjs-playlist-info').getText()).includes(text) }, { timeout: maxTime }) } @@ -42,7 +45,7 @@ export class PlayerPage { await browserSleep(2000) await browser.waitUntil(async () => { - return (await this.getWatchVideoPlayerCurrentTime()) >= 2 + return (await this.getWatchVideoPlayerCurrentTime()) >= waitUntilSec }) await videojsElem().click() diff --git a/client/e2e/src/po/video-list.po.ts b/client/e2e/src/po/video-list.po.ts new file mode 100644 index 000000000..f62c79adc --- /dev/null +++ b/client/e2e/src/po/video-list.po.ts @@ -0,0 +1,128 @@ +import { browserSleep, go } from '../utils' + +export class VideoListPage { + + constructor (private isMobileDevice: boolean, private isSafari: boolean) { + + } + + async goOnVideosList () { + let url: string + + // We did not upload a file on a mobile device + if (this.isMobileDevice === true || this.isSafari === true) { + url = 'https://peertube2.cpy.re/videos/local' + } else { + url = '/videos/recently-added' + } + + await go(url) + + // Waiting the following element does not work on Safari... + if (this.isSafari) return browserSleep(3000) + + await this.waitForList() + } + + async goOnLocal () { + await $('.menu-link[href="/videos/local"]').click() + await this.waitForTitle('Local videos') + } + + async goOnRecentlyAdded () { + await $('.menu-link[href="/videos/recently-added"]').click() + await this.waitForTitle('Recently added') + } + + async goOnTrending () { + await $('.menu-link[href="/videos/trending"]').click() + await this.waitForTitle('Trending') + } + + async goOnHomepage () { + await go('/home') + await this.waitForList() + } + + async goOnRootChannel () { + await go('/c/root_channel/videos') + await this.waitForList() + } + + async goOnRootAccount () { + await go('/a/root/videos') + await this.waitForList() + } + + async goOnRootAccountChannels () { + await go('/a/root/video-channels') + await this.waitForList() + } + + getNSFWFilter () { + return $$('.active-filter').filter(async a => { + return (await a.getText()).includes('Sensitive') + }).then(f => f[0]) + } + + async getVideosListName () { + const elems = await $$('.videos .video-miniature .video-miniature-name') + const texts = await Promise.all(elems.map(e => e.getText())) + + return texts.map(t => t.trim()) + } + + videoExists (name: string) { + return $('.video-miniature-name=' + name).isDisplayed() + } + + async videoIsBlurred (name: string) { + const filter = await $('.video-miniature-name=' + name).getCSSProperty('filter') + + return filter.value !== 'none' + } + + async clickOnVideo (videoName: string) { + const video = async () => { + const videos = await $$('.videos .video-miniature .video-miniature-name').filter(async e => { + const t = await e.getText() + + return t === videoName + }) + + return videos[0] + } + + await browser.waitUntil(async () => { + const elem = await video() + + return elem?.isClickable() + }); + + (await video()).click() + + await browser.waitUntil(async () => (await browser.getUrl()).includes('/w/')) + } + + async clickOnFirstVideo () { + const video = () => $('.videos .video-miniature .video-thumbnail') + const videoName = () => $('.videos .video-miniature .video-miniature-name') + + await video().waitForClickable() + + const textToReturn = await videoName().getText() + await video().click() + + await browser.waitUntil(async () => (await browser.getUrl()).includes('/w/')) + + return textToReturn + } + + private waitForList () { + return $('.videos .video-miniature .video-miniature-name').waitForDisplayed() + } + + private waitForTitle (title: string) { + return $('h1=' + title).waitForDisplayed() + } +} diff --git a/client/e2e/src/po/video-search.po.ts b/client/e2e/src/po/video-search.po.ts new file mode 100644 index 000000000..5446718d1 --- /dev/null +++ b/client/e2e/src/po/video-search.po.ts @@ -0,0 +1,11 @@ +export class VideoSearchPage { + + async search (search: string) { + await $('#search-video').setValue(search) + await $('my-header .icon-search').click() + + await browser.waitUntil(() => { + return $('my-video-miniature').isDisplayed() + }) + } +} diff --git a/client/e2e/src/po/video-upload.po.ts b/client/e2e/src/po/video-upload.po.ts index 34f916b55..dd437c390 100644 --- a/client/e2e/src/po/video-upload.po.ts +++ b/client/e2e/src/po/video-upload.po.ts @@ -1,4 +1,5 @@ import { join } from 'path' +import { clickOnCheckbox } from '../utils' export class VideoUploadPage { async navigateTo () { @@ -30,6 +31,10 @@ export class VideoUploadPage { }) } + setAsNSFW () { + return clickOnCheckbox('nsfw') + } + async validSecondUploadStep (videoName: string) { const nameInput = $('input#name') await nameInput.clearValue() diff --git a/client/e2e/src/po/video-watch.po.ts b/client/e2e/src/po/video-watch.po.ts index c07f4b25f..41425f4d7 100644 --- a/client/e2e/src/po/video-watch.po.ts +++ b/client/e2e/src/po/video-watch.po.ts @@ -1,37 +1,16 @@ -import { FIXTURE_URLS } from '../urls' -import { browserSleep, go } from '../utils' +import { browserSleep, FIXTURE_URLS, go } from '../utils' export class VideoWatchPage { - async goOnVideosList (isMobileDevice: boolean, isSafari: boolean) { - let url: string - - // We did not upload a file on a mobile device - if (isMobileDevice === true || isSafari === true) { - url = 'https://peertube2.cpy.re/videos/local' - } else { - url = '/videos/recently-added' - } - - await go(url) - - // Waiting the following element does not work on Safari... - if (isSafari) return browserSleep(3000) - await $('.videos .video-miniature .video-miniature-name').waitForDisplayed() - } - - async getVideosListName () { - const elems = await $$('.videos .video-miniature .video-miniature-name') - const texts = await Promise.all(elems.map(e => e.getText())) + constructor (private isMobileDevice: boolean, private isSafari: boolean) { - return texts.map(t => t.trim()) } - waitWatchVideoName (videoName: string, isMobileDevice: boolean, isSafari: boolean) { - if (isSafari) return browserSleep(5000) + waitWatchVideoName (videoName: string) { + if (this.isSafari) return browserSleep(5000) // On mobile we display the first node, on desktop the second - const index = isMobileDevice ? 0 : 1 + const index = this.isMobileDevice ? 0 : 1 return browser.waitUntil(async () => { return (await $$('.video-info .video-info-name')[index].getText()).includes(videoName) @@ -58,42 +37,6 @@ export class VideoWatchPage { return go(FIXTURE_URLS.HLS_PLAYLIST_EMBED) } - async clickOnVideo (videoName: string) { - const video = async () => { - const videos = await $$('.videos .video-miniature .video-miniature-name').filter(async e => { - const t = await e.getText() - - return t === videoName - }) - - return videos[0] - } - - await browser.waitUntil(async () => { - const elem = await video() - - return elem?.isClickable() - }); - - (await video()).click() - - await browser.waitUntil(async () => (await browser.getUrl()).includes('/w/')) - } - - async clickOnFirstVideo () { - const video = () => $('.videos .video-miniature .video-thumbnail') - const videoName = () => $('.videos .video-miniature .video-miniature-name') - - await video().waitForClickable() - - const textToReturn = await videoName().getText() - await video().click() - - await browser.waitUntil(async () => (await browser.getUrl()).includes('/w/')) - - return textToReturn - } - async clickOnUpdate () { const dropdown = $('my-video-actions-dropdown .action-button') await dropdown.click() diff --git a/client/e2e/src/suites-all/videos.e2e-spec.ts b/client/e2e/src/suites-all/videos.e2e-spec.ts new file mode 100644 index 000000000..3b8305a25 --- /dev/null +++ b/client/e2e/src/suites-all/videos.e2e-spec.ts @@ -0,0 +1,234 @@ +import { LoginPage } from '../po/login.po' +import { MyAccountPage } from '../po/my-account' +import { PlayerPage } from '../po/player.po' +import { VideoListPage } from '../po/video-list.po' +import { VideoUpdatePage } from '../po/video-update.po' +import { VideoUploadPage } from '../po/video-upload.po' +import { VideoWatchPage } from '../po/video-watch.po' +import { FIXTURE_URLS, go, isIOS, isMobileDevice, isSafari, waitServerUp } from '../utils' + +function isUploadUnsupported () { + if (isMobileDevice() || isSafari()) { + console.log('Skipping because we are on a real device or Safari and BrowserStack does not support file upload.') + return true + } + + return false +} + +describe('Videos all workflow', () => { + let videoWatchPage: VideoWatchPage + let videoListPage: VideoListPage + let videoUploadPage: VideoUploadPage + let videoUpdatePage: VideoUpdatePage + let myAccountPage: MyAccountPage + let loginPage: LoginPage + let playerPage: PlayerPage + + let videoName = Math.random() + ' video' + const video2Name = Math.random() + ' second video' + const playlistName = Math.random() + ' playlist' + let videoWatchUrl: string + + before(async () => { + if (isIOS()) { + console.log('iOS detected') + } else if (isMobileDevice()) { + console.log('Android detected.') + } else if (isSafari()) { + console.log('Safari detected.') + } + + if (isUploadUnsupported()) return + + await waitServerUp() + }) + + beforeEach(async () => { + videoWatchPage = new VideoWatchPage(isMobileDevice(), isSafari()) + videoUploadPage = new VideoUploadPage() + videoUpdatePage = new VideoUpdatePage() + myAccountPage = new MyAccountPage() + loginPage = new LoginPage() + playerPage = new PlayerPage() + videoListPage = new VideoListPage(isMobileDevice(), isSafari()) + + if (!isMobileDevice()) { + await browser.maximizeWindow() + } + }) + + it('Should log in', async () => { + if (isMobileDevice() || isSafari()) { + console.log('Skipping because we are on a real device or Safari and BrowserStack does not support file upload.') + return + } + + return loginPage.loginAsRootUser() + }) + + it('Should upload a video', async () => { + if (isUploadUnsupported()) return + + await videoUploadPage.navigateTo() + + await videoUploadPage.uploadVideo() + return videoUploadPage.validSecondUploadStep(videoName) + }) + + it('Should list videos', async () => { + await videoListPage.goOnVideosList() + + if (isUploadUnsupported()) return + + const videoNames = await videoListPage.getVideosListName() + expect(videoNames).toContain(videoName) + }) + + it('Should go on video watch page', async () => { + let videoNameToExcept = videoName + + if (isMobileDevice() || isSafari()) { + await go(FIXTURE_URLS.WEBTORRENT_VIDEO) + videoNameToExcept = 'E2E tests' + } else { + await videoListPage.clickOnVideo(videoName) + } + + return videoWatchPage.waitWatchVideoName(videoNameToExcept) + }) + + it('Should play the video', async () => { + videoWatchUrl = await browser.getUrl() + + await playerPage.playAndPauseVideo(true, 2) + expect(await playerPage.getWatchVideoPlayerCurrentTime()).toBeGreaterThanOrEqual(2) + }) + + it('Should watch the associated embed video', async () => { + await videoWatchPage.goOnAssociatedEmbed() + + await playerPage.playAndPauseVideo(false, 2) + expect(await playerPage.getWatchVideoPlayerCurrentTime()).toBeGreaterThanOrEqual(2) + }) + + it('Should watch the p2p media loader embed video', async () => { + await videoWatchPage.goOnP2PMediaLoaderEmbed() + + await playerPage.playAndPauseVideo(false, 2) + expect(await playerPage.getWatchVideoPlayerCurrentTime()).toBeGreaterThanOrEqual(2) + }) + + it('Should update the video', async () => { + if (isUploadUnsupported()) return + + await go(videoWatchUrl) + + await videoWatchPage.clickOnUpdate() + + videoName += ' updated' + await videoUpdatePage.updateName(videoName) + + await videoUpdatePage.validUpdate() + + const name = await videoWatchPage.getVideoName() + expect(name).toEqual(videoName) + }) + + it('Should add the video in my playlist', async () => { + if (isUploadUnsupported()) return + + await videoWatchPage.clickOnSave() + + await videoWatchPage.createPlaylist(playlistName) + + await videoWatchPage.saveToPlaylist(playlistName) + await browser.pause(5000) + + await videoUploadPage.navigateTo() + + await videoUploadPage.uploadVideo() + await videoUploadPage.validSecondUploadStep(video2Name) + + await videoWatchPage.clickOnSave() + await videoWatchPage.saveToPlaylist(playlistName) + }) + + it('Should have the playlist in my account', async () => { + if (isUploadUnsupported()) return + + await myAccountPage.navigateToMyPlaylists() + + const videosNumberText = await myAccountPage.getPlaylistVideosText(playlistName) + expect(videosNumberText).toEqual('2 videos') + + await myAccountPage.clickOnPlaylist(playlistName) + + const count = await myAccountPage.countTotalPlaylistElements() + expect(count).toEqual(2) + }) + + it('Should watch the playlist', async () => { + if (isUploadUnsupported()) return + + await myAccountPage.playPlaylist() + + await videoWatchPage.waitUntilVideoName(video2Name, 30 * 1000) + }) + + it('Should watch the webtorrent playlist in the embed', async () => { + if (isUploadUnsupported()) return + + const accessToken = await browser.execute(`return window.localStorage.getItem('access_token');`) + const refreshToken = await browser.execute(`return window.localStorage.getItem('refresh_token');`) + + await myAccountPage.goOnAssociatedPlaylistEmbed() + + await playerPage.waitUntilPlayerWrapper() + + console.log('Will set %s and %s tokens in local storage.', accessToken, refreshToken) + + await browser.execute(`window.localStorage.setItem('access_token', '${accessToken}');`) + await browser.execute(`window.localStorage.setItem('refresh_token', '${refreshToken}');`) + await browser.execute(`window.localStorage.setItem('token_type', 'Bearer');`) + + await browser.refresh() + + await playerPage.playVideo() + + await playerPage.waitUntilPlaylistInfo('2/2', 30 * 1000) + }) + + it('Should watch the HLS playlist in the embed', async () => { + await videoWatchPage.goOnP2PMediaLoaderPlaylistEmbed() + + await playerPage.playVideo() + + await playerPage.waitUntilPlaylistInfo('2/2', 30 * 1000) + }) + + it('Should delete the video 2', async () => { + if (isUploadUnsupported()) return + + // Go to the dev website + await go(videoWatchUrl) + + await myAccountPage.navigateToMyVideos() + + await myAccountPage.removeVideo(video2Name) + await myAccountPage.validRemove() + + await browser.waitUntil(async () => { + const count = await myAccountPage.countVideos([ videoName, video2Name ]) + + return count === 1 + }) + }) + + it('Should delete the first video', async () => { + if (isUploadUnsupported()) return + + await myAccountPage.removeVideo(videoName) + await myAccountPage.validRemove() + }) +}) diff --git a/client/e2e/src/suites-local/videos-list.e2e-spec.ts b/client/e2e/src/suites-local/videos-list.e2e-spec.ts new file mode 100644 index 000000000..1e0a88859 --- /dev/null +++ b/client/e2e/src/suites-local/videos-list.e2e-spec.ts @@ -0,0 +1,195 @@ +import { AdminConfigPage } from '../po/admin-config.po' +import { LoginPage } from '../po/login.po' +import { MyAccountPage } from '../po/my-account' +import { VideoListPage } from '../po/video-list.po' +import { VideoSearchPage } from '../po/video-search.po' +import { VideoUploadPage } from '../po/video-upload.po' +import { NSFWPolicy } from '../types/common' +import { isMobileDevice, isSafari, waitServerUp } from '../utils' + +describe('Videos list', () => { + let videoListPage: VideoListPage + let videoUploadPage: VideoUploadPage + let adminConfigPage: AdminConfigPage + let loginPage: LoginPage + let myAccountPage: MyAccountPage + let videoSearchPage: VideoSearchPage + + const seed = Math.random() + const nsfwVideo = seed + ' - nsfw' + const normalVideo = seed + ' - normal' + + async function checkNormalVideo () { + expect(await videoListPage.videoExists(normalVideo)).toBeTruthy() + expect(await videoListPage.videoIsBlurred(normalVideo)).toBeFalsy() + } + + async function checkNSFWVideo (policy: NSFWPolicy, filterText?: string) { + if (policy === 'do_not_list') { + if (filterText) expect(filterText).toContain('hidden') + + expect(await videoListPage.videoExists(nsfwVideo)).toBeFalsy() + return + } + + if (policy === 'blur') { + if (filterText) expect(filterText).toContain('blurred') + + expect(await videoListPage.videoExists(nsfwVideo)).toBeTruthy() + expect(await videoListPage.videoIsBlurred(nsfwVideo)).toBeTruthy() + return + } + + // display + if (filterText) expect(filterText).toContain('displayed') + + expect(await videoListPage.videoExists(nsfwVideo)).toBeTruthy() + expect(await videoListPage.videoIsBlurred(nsfwVideo)).toBeFalsy() + } + + async function checkCommonVideoListPages (policy: NSFWPolicy) { + const promisesWithFilters = [ + videoListPage.goOnRootAccount, + videoListPage.goOnLocal, + videoListPage.goOnRecentlyAdded, + videoListPage.goOnTrending, + videoListPage.goOnRootChannel + ] + + for (const p of promisesWithFilters) { + await p.call(videoListPage) + + const filter = await videoListPage.getNSFWFilter() + const filterText = await filter.getText() + + await checkNormalVideo() + await checkNSFWVideo(policy, filterText) + } + + const promisesWithoutFilters = [ + videoListPage.goOnRootAccountChannels, + videoListPage.goOnHomepage + ] + for (const p of promisesWithoutFilters) { + await p.call(videoListPage) + + await checkNormalVideo() + await checkNSFWVideo(policy) + } + } + + async function checkSearchPage (policy: NSFWPolicy) { + await videoSearchPage.search(normalVideo) + await checkNormalVideo() + + await videoSearchPage.search(nsfwVideo) + await checkNSFWVideo(policy) + } + + async function updateAdminNSFW (nsfw: NSFWPolicy) { + await adminConfigPage.navigateTo('instance-information') + await adminConfigPage.updateNSFWSetting(nsfw) + await adminConfigPage.save() + } + + async function updateUserNSFW (nsfw: NSFWPolicy) { + await myAccountPage.navigateToMySettings() + await myAccountPage.updateNSFW(nsfw) + } + + before(async () => { + await waitServerUp() + }) + + beforeEach(async () => { + videoListPage = new VideoListPage(isMobileDevice(), isSafari()) + adminConfigPage = new AdminConfigPage() + loginPage = new LoginPage() + videoUploadPage = new VideoUploadPage() + myAccountPage = new MyAccountPage() + videoSearchPage = new VideoSearchPage() + + await browser.maximizeWindow() + }) + + it('Should login and disable NSFW', async () => { + await loginPage.loginAsRootUser() + await updateUserNSFW('display') + }) + + it('Should set the homepage', async () => { + await adminConfigPage.navigateTo('instance-homepage') + await adminConfigPage.updateHomepage('') + await adminConfigPage.save() + }) + + it('Should upload 2 videos (NSFW and classic videos)', async () => { + await videoUploadPage.navigateTo() + await videoUploadPage.uploadVideo() + await videoUploadPage.setAsNSFW() + await videoUploadPage.validSecondUploadStep(nsfwVideo) + + await videoUploadPage.navigateTo() + await videoUploadPage.uploadVideo() + await videoUploadPage.validSecondUploadStep(normalVideo) + }) + + it('Should logout', async function () { + await loginPage.logout() + }) + + describe('Anonymous users', function () { + + it('Should correctly handle do not list', async () => { + await loginPage.loginAsRootUser() + await updateAdminNSFW('do_not_list') + + await loginPage.logout() + await checkCommonVideoListPages('do_not_list') + await checkSearchPage('do_not_list') + }) + + it('Should correctly handle blur', async () => { + await loginPage.loginAsRootUser() + await updateAdminNSFW('blur') + + await loginPage.logout() + await checkCommonVideoListPages('blur') + await checkSearchPage('blur') + }) + + it('Should correctly handle display', async () => { + await loginPage.loginAsRootUser() + await updateAdminNSFW('display') + + await loginPage.logout() + await checkCommonVideoListPages('display') + await checkSearchPage('display') + }) + }) + + describe('Logged in users', function () { + + before(async () => { + await loginPage.loginAsRootUser() + }) + + it('Should correctly handle do not list', async () => { + await updateUserNSFW('do_not_list') + await checkCommonVideoListPages('do_not_list') + await checkSearchPage('do_not_list') + }) + + it('Should correctly handle blur', async () => { + await updateUserNSFW('blur') + await checkCommonVideoListPages('blur') + await checkSearchPage('blur') + }) + + it('Should correctly handle display', async () => { + await updateUserNSFW('display') + await checkCommonVideoListPages('display') + await checkSearchPage('display') + }) + }) +}) diff --git a/client/e2e/src/types/common.ts b/client/e2e/src/types/common.ts new file mode 100644 index 000000000..c0b59d297 --- /dev/null +++ b/client/e2e/src/types/common.ts @@ -0,0 +1 @@ +export type NSFWPolicy = 'do_not_list' | 'blur' | 'display' diff --git a/client/e2e/src/urls.ts b/client/e2e/src/urls.ts deleted file mode 100644 index 664c65931..000000000 --- a/client/e2e/src/urls.ts +++ /dev/null @@ -1,10 +0,0 @@ -const FIXTURE_URLS = { - WEBTORRENT_VIDEO: 'https://peertube2.cpy.re/w/122d093a-1ede-43bd-bd34-59d2931ffc5e', - - HLS_EMBED: 'https://peertube2.cpy.re/videos/embed/969bf103-7818-43b5-94a0-de159e13de50', - HLS_PLAYLIST_EMBED: 'https://peertube2.cpy.re/video-playlists/embed/73804a40-da9a-40c2-b1eb-2c6d9eec8f0a' -} - -export { - FIXTURE_URLS -} diff --git a/client/e2e/src/utils.ts b/client/e2e/src/utils.ts deleted file mode 100644 index df1c29238..000000000 --- a/client/e2e/src/utils.ts +++ /dev/null @@ -1,37 +0,0 @@ -async function browserSleep (amount: number) { - await browser.pause(amount) -} - -function isMobileDevice () { - const platformName = (browser.capabilities['platformName'] || '').toLowerCase() - - return platformName === 'android' || platformName === 'ios' -} - -function isSafari () { - return browser.capabilities['browserName'] && - browser.capabilities['browserName'].toLowerCase() === 'safari' -} - -function isIOS () { - return isMobileDevice() && isSafari() -} - -async function go (url: string) { - await browser.url(url) - - // Hide notifications that could fail tests when hiding buttons - await browser.execute(() => { - const style = document.createElement('style') - style.innerHTML = 'p-toast { display: none }' - document.head.appendChild(style) - }) -} - -export { - isMobileDevice, - isSafari, - isIOS, - go, - browserSleep -} diff --git a/client/e2e/src/utils/common.ts b/client/e2e/src/utils/common.ts new file mode 100644 index 000000000..eb5f6b450 --- /dev/null +++ b/client/e2e/src/utils/common.ts @@ -0,0 +1,47 @@ +async function browserSleep (amount: number) { + await browser.pause(amount) +} + +function isMobileDevice () { + const platformName = (browser.capabilities['platformName'] || '').toLowerCase() + + return platformName === 'android' || platformName === 'ios' +} + +function isSafari () { + return browser.capabilities['browserName'] && + browser.capabilities['browserName'].toLowerCase() === 'safari' +} + +function isIOS () { + return isMobileDevice() && isSafari() +} + +async function go (url: string) { + await browser.url(url) + + // Hide notifications that could fail tests when hiding buttons + await browser.execute(() => { + const style = document.createElement('style') + style.innerHTML = 'p-toast { display: none }' + document.head.appendChild(style) + }) +} + +async function waitServerUp () { + await browser.waitUntil(async () => { + await go('/') + await browserSleep(500) + + return $('').isDisplayed() + }, { timeout: 20 * 1000 }) +} + +export { + isMobileDevice, + isSafari, + isIOS, + waitServerUp, + go, + browserSleep +} diff --git a/client/e2e/src/utils/elements.ts b/client/e2e/src/utils/elements.ts new file mode 100644 index 000000000..cadc46cce --- /dev/null +++ b/client/e2e/src/utils/elements.ts @@ -0,0 +1,7 @@ +function clickOnCheckbox (name: string) { + return $(`my-peertube-checkbox[inputname=${name}] label`).click() +} + +export { + clickOnCheckbox +} diff --git a/client/e2e/src/utils/index.ts b/client/e2e/src/utils/index.ts new file mode 100644 index 000000000..5da1ad517 --- /dev/null +++ b/client/e2e/src/utils/index.ts @@ -0,0 +1,3 @@ +export * from './common' +export * from './elements' +export * from './urls' diff --git a/client/e2e/src/utils/urls.ts b/client/e2e/src/utils/urls.ts new file mode 100644 index 000000000..664c65931 --- /dev/null +++ b/client/e2e/src/utils/urls.ts @@ -0,0 +1,10 @@ +const FIXTURE_URLS = { + WEBTORRENT_VIDEO: 'https://peertube2.cpy.re/w/122d093a-1ede-43bd-bd34-59d2931ffc5e', + + HLS_EMBED: 'https://peertube2.cpy.re/videos/embed/969bf103-7818-43b5-94a0-de159e13de50', + HLS_PLAYLIST_EMBED: 'https://peertube2.cpy.re/video-playlists/embed/73804a40-da9a-40c2-b1eb-2c6d9eec8f0a' +} + +export { + FIXTURE_URLS +} diff --git a/client/e2e/src/videos.e2e-spec.ts b/client/e2e/src/videos.e2e-spec.ts deleted file mode 100644 index e09e8c675..000000000 --- a/client/e2e/src/videos.e2e-spec.ts +++ /dev/null @@ -1,237 +0,0 @@ -import { LoginPage } from './po/login.po' -import { MyAccountPage } from './po/my-account' -import { PlayerPage } from './po/player.po' -import { VideoUpdatePage } from './po/video-update.po' -import { VideoUploadPage } from './po/video-upload.po' -import { VideoWatchPage } from './po/video-watch.po' -import { FIXTURE_URLS } from './urls' -import { browserSleep, go, isIOS, isMobileDevice, isSafari } from './utils' - -function isUploadUnsupported () { - if (isMobileDevice() || isSafari()) { - console.log('Skipping because we are on a real device or Safari and BrowserStack does not support file upload.') - return true - } - - return false -} - -describe('Videos workflow', () => { - let videoWatchPage: VideoWatchPage - let videoUploadPage: VideoUploadPage - let videoUpdatePage: VideoUpdatePage - let myAccountPage: MyAccountPage - let loginPage: LoginPage - let playerPage: PlayerPage - - let videoName = Math.random() + ' video' - const video2Name = Math.random() + ' second video' - const playlistName = Math.random() + ' playlist' - let videoWatchUrl: string - - before(async () => { - if (isIOS()) { - console.log('iOS detected') - } else if (isMobileDevice()) { - console.log('Android detected.') - } else if (isSafari()) { - console.log('Safari detected.') - } - - if (isUploadUnsupported()) return - - await browser.waitUntil(async () => { - await go('/') - await browserSleep(500) - - return $('').isDisplayed() - }, { timeout: 20 * 1000 }) - }) - - beforeEach(async () => { - videoWatchPage = new VideoWatchPage() - videoUploadPage = new VideoUploadPage() - videoUpdatePage = new VideoUpdatePage() - myAccountPage = new MyAccountPage() - loginPage = new LoginPage() - playerPage = new PlayerPage() - - if (!isMobileDevice()) { - await browser.maximizeWindow() - } - }) - - it('Should log in', async () => { - if (isMobileDevice() || isSafari()) { - console.log('Skipping because we are on a real device or Safari and BrowserStack does not support file upload.') - return - } - - return loginPage.loginAsRootUser() - }) - - it('Should upload a video', async () => { - if (isUploadUnsupported()) return - - await videoUploadPage.navigateTo() - - await videoUploadPage.uploadVideo() - return videoUploadPage.validSecondUploadStep(videoName) - }) - - it('Should list videos', async () => { - await videoWatchPage.goOnVideosList(isMobileDevice(), isSafari()) - - if (isUploadUnsupported()) return - - const videoNames = await videoWatchPage.getVideosListName() - expect(videoNames).toContain(videoName) - }) - - it('Should go on video watch page', async () => { - let videoNameToExcept = videoName - - if (isMobileDevice() || isSafari()) { - await go(FIXTURE_URLS.WEBTORRENT_VIDEO) - videoNameToExcept = 'E2E tests' - } else { - await videoWatchPage.clickOnVideo(videoName) - } - - return videoWatchPage.waitWatchVideoName(videoNameToExcept, isMobileDevice(), isSafari()) - }) - - it('Should play the video', async () => { - videoWatchUrl = await browser.getUrl() - - await playerPage.playAndPauseVideo(true, 2) - expect(await playerPage.getWatchVideoPlayerCurrentTime()).toBeGreaterThanOrEqual(2) - }) - - it('Should watch the associated embed video', async () => { - await videoWatchPage.goOnAssociatedEmbed() - - await playerPage.playAndPauseVideo(false, 2) - expect(await playerPage.getWatchVideoPlayerCurrentTime()).toBeGreaterThanOrEqual(2) - }) - - it('Should watch the p2p media loader embed video', async () => { - await videoWatchPage.goOnP2PMediaLoaderEmbed() - - await playerPage.playAndPauseVideo(false, 2) - expect(await playerPage.getWatchVideoPlayerCurrentTime()).toBeGreaterThanOrEqual(2) - }) - - it('Should update the video', async () => { - if (isUploadUnsupported()) return - - await go(videoWatchUrl) - - await videoWatchPage.clickOnUpdate() - - videoName += ' updated' - await videoUpdatePage.updateName(videoName) - - await videoUpdatePage.validUpdate() - - const name = await videoWatchPage.getVideoName() - expect(name).toEqual(videoName) - }) - - it('Should add the video in my playlist', async () => { - if (isUploadUnsupported()) return - - await videoWatchPage.clickOnSave() - - await videoWatchPage.createPlaylist(playlistName) - - await videoWatchPage.saveToPlaylist(playlistName) - await browser.pause(5000) - - await videoUploadPage.navigateTo() - - await videoUploadPage.uploadVideo() - await videoUploadPage.validSecondUploadStep(video2Name) - - await videoWatchPage.clickOnSave() - await videoWatchPage.saveToPlaylist(playlistName) - }) - - it('Should have the playlist in my account', async () => { - if (isUploadUnsupported()) return - - await myAccountPage.navigateToMyPlaylists() - - const videosNumberText = await myAccountPage.getPlaylistVideosText(playlistName) - expect(videosNumberText).toEqual('2 videos') - - await myAccountPage.clickOnPlaylist(playlistName) - - const count = await myAccountPage.countTotalPlaylistElements() - expect(count).toEqual(2) - }) - - it('Should watch the playlist', async () => { - if (isUploadUnsupported()) return - - await myAccountPage.playPlaylist() - - await videoWatchPage.waitUntilVideoName(video2Name, 30 * 1000) - }) - - it('Should watch the webtorrent playlist in the embed', async () => { - if (isUploadUnsupported()) return - - const accessToken = await browser.execute(`return window.localStorage.getItem('access_token');`) - const refreshToken = await browser.execute(`return window.localStorage.getItem('refresh_token');`) - - await myAccountPage.goOnAssociatedPlaylistEmbed() - - await playerPage.waitUntilPlayerWrapper() - - console.log('Will set %s and %s tokens in local storage.', accessToken, refreshToken) - - await browser.execute(`window.localStorage.setItem('access_token', '${accessToken}');`) - await browser.execute(`window.localStorage.setItem('refresh_token', '${refreshToken}');`) - await browser.execute(`window.localStorage.setItem('token_type', 'Bearer');`) - - await browser.refresh() - - await playerPage.playVideo() - - await playerPage.waitUntilPlaylistInfo('2/2', 30 * 1000) - }) - - it('Should watch the HLS playlist in the embed', async () => { - await videoWatchPage.goOnP2PMediaLoaderPlaylistEmbed() - - await playerPage.playVideo() - - await playerPage.waitUntilPlaylistInfo('2/2', 30 * 1000) - }) - - it('Should delete the video 2', async () => { - if (isUploadUnsupported()) return - - // Go to the dev website - await go(videoWatchUrl) - - await myAccountPage.navigateToMyVideos() - - await myAccountPage.removeVideo(video2Name) - await myAccountPage.validRemove() - - await browser.waitUntil(async () => { - const count = await myAccountPage.countVideos([ videoName, video2Name ]) - - return count === 1 - }) - }) - - it('Should delete the first video', async () => { - if (isUploadUnsupported()) return - - await myAccountPage.removeVideo(videoName) - await myAccountPage.validRemove() - }) -}) -- cgit v1.2.3