diff --git a/.github/workflows/playwright.yml b/.github/workflows/playwright.yml new file mode 100644 index 000000000..443b08a31 --- /dev/null +++ b/.github/workflows/playwright.yml @@ -0,0 +1,42 @@ +name: Playwright Tests + +on: + push: + branches: + - main + - development + pull_request: + branches: + - main + - development + +env: + # override value from .env.testnet used in CI + NEXT_PUBLIC_SITE_URL: http://127.0.0.1:3000 + +jobs: + test: + timeout-minutes: 60 + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 + with: + node-version: lts/* + cache: yarn + - name: Install packages + run: yarn install + - name: Install Playwright browsers + run: yarn playwright install --with-deps + - name: Set up testnet environment + run: cp apps/dapp/.env.testnet apps/dapp/.env.local + - name: Build dApp + run: yarn dapp build + - name: Run Playwright tests + run: yarn test:e2e + - uses: actions/upload-artifact@v4 + if: always() + with: + name: playwright-report + path: apps/dapp/playwright-report/ + retention-days: 30 diff --git a/.gitignore b/.gitignore index 0aa576f5d..02259aa85 100644 --- a/.gitignore +++ b/.gitignore @@ -15,10 +15,7 @@ node_modules/ out/ .env -.env.development.local -.env.local -.env.production.local -.env.test.local +.env*.local *.swo *.swp @@ -28,3 +25,9 @@ out/ \#*\# .turbo + +# playwright +test-results/ +playwright-report/ +blob-report/ +playwright/.cache/ diff --git a/apps/dapp/package.json b/apps/dapp/package.json index 9eb1e36dd..f24982fbf 100644 --- a/apps/dapp/package.json +++ b/apps/dapp/package.json @@ -15,6 +15,7 @@ "start": "next start", "test": "jest --passWithNoTests", "test:watch": "jest --watch", + "test:e2e": "cp .env.testnet .env.local && playwright test", "ts": "tsc --noEmit --incremental", "ts:watch": "ts --watch" }, @@ -53,10 +54,12 @@ "devDependencies": { "@dao-dao/config": "2.4.0-rc.8", "@next/bundle-analyzer": "^14.1.0", + "@playwright/test": "^1.44.1", "@sentry/cli": "^2.21.3", "@solana/web3.js": "^1.75.0", "@types/cors": "^2.8.13", "@types/lodash.clonedeep": "^4.5.0", + "@types/node": "^20.14.2", "@types/react": "^17.0.37", "autoprefixer": "^9.8.6", "eslint": "^8.23.1", diff --git a/apps/dapp/playwright.config.ts b/apps/dapp/playwright.config.ts new file mode 100644 index 000000000..7199867a2 --- /dev/null +++ b/apps/dapp/playwright.config.ts @@ -0,0 +1,80 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. +import { defineConfig, devices } from '@playwright/test' + +/** + * See https://playwright.dev/docs/test-configuration. + */ +export default defineConfig({ + testDir: './tests', + /* Run tests in files in parallel */ + fullyParallel: true, + /* Fail the build on CI if you accidentally left test.only in the source code. */ + forbidOnly: !!process.env.CI, + /* Retry on CI only */ + retries: process.env.CI ? 2 : 0, + /* Opt out of parallel tests on CI. */ + // workers: process.env.CI ? 1 : undefined, + /* Reporter to use. See https://playwright.dev/docs/test-reporters */ + reporter: 'html', + /* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */ + use: { + /* Base URL to use in actions like `await page.goto('/')`. */ + baseURL: 'http://127.0.0.1:3000', + + /* In CI, store all traces for analysis. Locally, just use failed traces. */ + trace: process.env.CI ? 'on' : 'retain-on-failure', + }, + + /* Configure projects for major browsers */ + projects: [ + { + name: 'chromium', + use: { ...devices['Desktop Chrome'] }, + }, + + { + name: 'firefox', + use: { ...devices['Desktop Firefox'] }, + }, + + { + name: 'webkit', + use: { ...devices['Desktop Safari'] }, + }, + + /* Test against mobile viewports. */ + // { + // name: 'Mobile Chrome', + // use: { ...devices['Pixel 5'] }, + // }, + // { + // name: 'Mobile Safari', + // use: { ...devices['iPhone 12'] }, + // }, + + /* Test against branded browsers. */ + // { + // name: 'Microsoft Edge', + // use: { ...devices['Desktop Edge'], channel: 'msedge' }, + // }, + // { + // name: 'Google Chrome', + // use: { ...devices['Desktop Chrome'], channel: 'chrome' }, + // }, + ], + + /* Build and run server before starting the tests */ + webServer: { + command: 'yarn start', + url: 'http://127.0.0.1:3000', + reuseExistingServer: !process.env.CI, + // allow 10 minutes for the server to build + timeout: 10 * 60 * 1000, + }, + + expect: { + // Default expect timeout to 20 seconds. + timeout: 20 * 1000, + }, +}) diff --git a/apps/dapp/tests/chain.spec.ts b/apps/dapp/tests/chain.spec.ts new file mode 100644 index 000000000..0510c6038 --- /dev/null +++ b/apps/dapp/tests/chain.spec.ts @@ -0,0 +1,30 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. + +import { expect, test } from '@playwright/test' + +import './setup' + +test('chain home/proposals tab renders', async ({ page }) => { + await page.goto('/dao/juno/proposals') + + // Expect description to exist. + await expect( + page.getByText('Native chain governance for Juno Testnet.') + ).toBeVisible() + + // Expect "New proposal" button to exist. + await expect(page.getByText('New proposal', { exact: true })).toBeVisible() +}) + +test('chain treasury tab renders', async ({ page }) => { + await page.goto('/dao/juno/treasury') + + // Expect description to exist. + await expect( + page.getByText('Native chain governance for Juno Testnet.') + ).toBeVisible() + + // Expect "Token" title to exist. + await expect(page.getByText('Token', { exact: true })).toBeVisible() +}) diff --git a/apps/dapp/tests/create.spec.ts b/apps/dapp/tests/create.spec.ts new file mode 100644 index 000000000..5e657bcb9 --- /dev/null +++ b/apps/dapp/tests/create.spec.ts @@ -0,0 +1,21 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. + +import { expect, test } from '@playwright/test' + +import './setup' + +test('create DAO page renders', async ({ page }) => { + await page.goto('/dao/create') + + // Expect "New DAO" header to exist. + await expect( + page.locator('div.header-text').filter({ hasText: 'New DAO' }).first() + ).toBeVisible() + + // Expect "Log in" button to exist. + await expect(page.getByRole('button', { name: 'Log in' })).toBeVisible() + + // Expect "Continue" button to exist. + await expect(page.getByRole('button', { name: 'Continue' })).toBeVisible() +}) diff --git a/apps/dapp/tests/dao.spec.ts b/apps/dapp/tests/dao.spec.ts new file mode 100644 index 000000000..bff3743db --- /dev/null +++ b/apps/dapp/tests/dao.spec.ts @@ -0,0 +1,85 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. + +import { expect, test } from '@playwright/test' + +import './setup' + +test('DAO home tab renders', async ({ page }) => { + await page.goto( + '/dao/juno1vh0xndu9pj8g0lat6k3500mxusfduh804sf9hj7jpt4kgj0gmreq3jmqj4' + ) + + // Expect description to exist. + await expect(page.getByText('Worship the moon.')).toBeVisible() +}) + +test('DAO proposals tab renders', async ({ page }) => { + await page.goto( + '/dao/juno1vh0xndu9pj8g0lat6k3500mxusfduh804sf9hj7jpt4kgj0gmreq3jmqj4/proposals' + ) + + // Expect description to exist. + await expect(page.getByText('Worship the moon.')).toBeVisible() + + // Expect "New proposal" button to exist. + await expect(page.getByText('New proposal', { exact: true })).toBeVisible() +}) + +test('DAO treasury tab renders', async ({ page }) => { + await page.goto( + '/dao/juno1vh0xndu9pj8g0lat6k3500mxusfduh804sf9hj7jpt4kgj0gmreq3jmqj4/treasury' + ) + + // Expect no 404 error. + await expect(page.getByText('404: Not Found')).not.toBeVisible({ + timeout: 1000, + }) + + // Expect description to exist. + await expect(page.getByText('Worship the moon.')).toBeVisible() + + // Expect "Copy address" button to exist. + await expect(page.getByText('Copy address', { exact: true })).toBeVisible() + + // Expect "Tokens" title to exist. + await expect(page.getByText('Tokens', { exact: true })).toBeVisible() +}) + +test('DAO subDAOs tab renders', async ({ page }) => { + await page.goto( + '/dao/juno1vh0xndu9pj8g0lat6k3500mxusfduh804sf9hj7jpt4kgj0gmreq3jmqj4/subdaos' + ) + + // Expect description to exist. + await expect(page.getByText('Worship the moon.')).toBeVisible() + + // Expect "New SubDAO" button to exist. + await expect(page.getByText('New SubDAO', { exact: true })).toBeVisible() +}) + +test('DAO members tab renders', async ({ page }) => { + await page.goto( + '/dao/juno1vh0xndu9pj8g0lat6k3500mxusfduh804sf9hj7jpt4kgj0gmreq3jmqj4/members' + ) + + // Expect description to exist. + await expect(page.getByText('Worship the moon.')).toBeVisible() + + // Expect member voting power title to exist. + await expect( + page.getByText('Voting power', { exact: true }).first() + ).toBeVisible() +}) + +test('DAO apps tab renders', async ({ page }) => { + await page.goto( + '/dao/juno1vh0xndu9pj8g0lat6k3500mxusfduh804sf9hj7jpt4kgj0gmreq3jmqj4/apps' + ) + + // Expect description to exist. + await expect(page.getByText('Worship the moon.')).toBeVisible() + + // Expect member voting power title to exist. + await expect(page.getByRole('button', { name: 'Go' }).first()).toBeVisible() +}) diff --git a/apps/dapp/tests/home.spec.ts b/apps/dapp/tests/home.spec.ts new file mode 100644 index 000000000..d52576165 --- /dev/null +++ b/apps/dapp/tests/home.spec.ts @@ -0,0 +1,36 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. + +import { expect, test } from '@playwright/test' + +import './setup' + +test('home page renders', async ({ page }) => { + await page.goto('/') + + // Expect "Log in" button to exist. + await expect(page.getByRole('button', { name: 'Log in' })).toBeVisible() + + // Expect "Chain governance" title to exist. + await expect( + page.getByText('Chain governance', { exact: true }) + ).toBeVisible() + + // Expect "Featured DAOs" title to exist. + await expect(page.getByText('Featured DAOs', { exact: true })).toBeVisible() +}) + +test('chain-specific home page renders', async ({ page }) => { + await page.goto('/juno') + + // Expect "Log in" button to exist. + await expect(page.getByRole('button', { name: 'Log in' })).toBeVisible() + + // Expect "Chain governance" title to exist. + await expect( + page.getByText('Chain governance', { exact: true }) + ).toBeVisible() + + // Expect "Featured DAOs" title to exist. + await expect(page.getByText('Featured DAOs', { exact: true })).toBeVisible() +}) diff --git a/apps/dapp/tests/setup.ts b/apps/dapp/tests/setup.ts new file mode 100644 index 000000000..55beac57e --- /dev/null +++ b/apps/dapp/tests/setup.ts @@ -0,0 +1,16 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. + +/** + * Set up tests. + */ + +import { test } from '@playwright/test' + +test.beforeEach(async ({ page }) => { + await page.addInitScript(() => { + // Mark beta warning as approved so it doesn't appear and block the rest of + // the page. + window.localStorage.setItem('betaWarningAccepted', 'true') + }) +}) diff --git a/apps/dapp/tests/status.spec.ts b/apps/dapp/tests/status.spec.ts new file mode 100644 index 000000000..580c94f34 --- /dev/null +++ b/apps/dapp/tests/status.spec.ts @@ -0,0 +1,15 @@ +// GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. +// See the "LICENSE" file in the root directory of this package for more copyright information. + +import { expect, test } from '@playwright/test' + +import './setup' + +test('status page renders', async ({ page }) => { + await page.goto('/status') + + // Expect "Status" header to exist. + await expect( + page.locator('p.header-text').filter({ hasText: 'Status' }) + ).toBeVisible() +}) diff --git a/jest.config.js b/jest.config.js index e75538534..407d8a813 100644 --- a/jest.config.js +++ b/jest.config.js @@ -1,10 +1,15 @@ // GNU AFFERO GENERAL PUBLIC LICENSE Version 3. Copyright (C) 2022 DAO DAO Contributors. // See the "LICENSE" file in the root directory of this package for more copyright information. -/** @type {import('ts-jest/dist/types').InitialOptionsTsJest} */ +/** @type {import('ts-jest/dist/types').JestConfigWithTsJest} */ module.exports = { moduleDirectories: ['node_modules', ''], - testPathIgnorePatterns: ['/.next/', '/node_modules/'], + testPathIgnorePatterns: [ + '/.next/', + '/node_modules/', + // playwright e2e tests, not jest. jest unit tests are spread throughout + '/apps/dapp/tests/' + ], preset: 'ts-jest', testEnvironment: 'node', } diff --git a/package.json b/package.json index 192520cc4..55754aa47 100644 --- a/package.json +++ b/package.json @@ -13,6 +13,7 @@ "format": "prettier --write \"**/*.{css,json,md}\" && turbo run format --continue --parallel", "lint": "prettier --check \"**/*.{css,json,md}\" && turbo run lint --continue --parallel", "test": "jest", + "test:e2e": "yarn dapp test:e2e", "dapp": "yarn workspace @dao-dao/dapp", "sda": "yarn workspace @dao-dao/sda", "i18n": "yarn workspace @dao-dao/i18n", diff --git a/packages/stateful/server/makeGetDaoStaticProps.ts b/packages/stateful/server/makeGetDaoStaticProps.ts index 14cb23db0..0d7bc886f 100644 --- a/packages/stateful/server/makeGetDaoStaticProps.ts +++ b/packages/stateful/server/makeGetDaoStaticProps.ts @@ -26,7 +26,6 @@ import { } from '@dao-dao/types' import { cosmos } from '@dao-dao/types/protobuf' import { - CI, DAO_CORE_ACCENT_ITEM_KEY, DAO_STATIC_PROPS_CACHE_SECONDS, LEGACY_DAO_CONTRACT_NAMES, @@ -91,11 +90,6 @@ export class LegacyDaoError extends Error { export const makeGetDaoStaticProps: GetDaoStaticPropsMaker = ({ appMode, coreAddress: _coreAddress, getProps }) => async (context) => { - // Don't query chain if running in CI. - if (CI) { - return { notFound: true } - } - // Load server translations and get T function for use in getProps. const { i18nProps, serverT } = await serverSideTranslationsWithServerT( context.locale, diff --git a/yarn.lock b/yarn.lock index 6baedf888..b0985b4b9 100644 --- a/yarn.lock +++ b/yarn.lock @@ -6042,6 +6042,13 @@ resolved "https://registry.yarnpkg.com/@pkgjs/parseargs/-/parseargs-0.11.0.tgz#a77ea742fab25775145434eb1d2328cf5013ac33" integrity sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg== +"@playwright/test@^1.44.1": + version "1.44.1" + resolved "https://registry.yarnpkg.com/@playwright/test/-/test-1.44.1.tgz#cc874ec31342479ad99838040e99b5f604299bcb" + integrity sha512-1hZ4TNvD5z9VuhNJ/walIjvMVvYkZKf71axoF/uiAqpntQJXpG64dlXhoDXE3OczPuTuvjf/M5KWFg5VAVUS3Q== + dependencies: + playwright "1.44.1" + "@pmmmwh/react-refresh-webpack-plugin@^0.5.3": version "0.5.7" resolved "https://registry.yarnpkg.com/@pmmmwh/react-refresh-webpack-plugin/-/react-refresh-webpack-plugin-0.5.7.tgz#58f8217ba70069cc6a73f5d7e05e85b458c150e2" @@ -8325,6 +8332,13 @@ resolved "https://registry.yarnpkg.com/@types/node/-/node-14.18.36.tgz#c414052cb9d43fab67d679d5f3c641be911f5835" integrity sha512-FXKWbsJ6a1hIrRxv+FoukuHnGTgEzKYGi7kilfMae96AL9UNkPFNWJEEYWzdRI9ooIkbr4AKldyuSTLql06vLQ== +"@types/node@^20.14.2": + version "20.14.2" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.14.2.tgz#a5f4d2bcb4b6a87bffcaa717718c5a0f208f4a18" + integrity sha512-xyu6WAMVwv6AKFLB+e/7ySZVr/0zLCzOa7rSpq6jNwpqOrUbcACDWC+53d4n2QHOnDou0fbIsg8wZu/sxrnI4Q== + dependencies: + undici-types "~5.26.4" + "@types/normalize-package-data@^2.4.0": version "2.4.1" resolved "https://registry.yarnpkg.com/@types/normalize-package-data/-/normalize-package-data-2.4.1.tgz#d3357479a0fdfdd5907fe67e17e0a85c906e1301" @@ -14453,6 +14467,11 @@ fs.realpath@^1.0.0: resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== +fsevents@2.3.2, fsevents@^2.1.2, fsevents@^2.3.2: + version "2.3.2" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a" + integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA== + fsevents@^1.2.7: version "1.2.13" resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-1.2.13.tgz#f325cb0455592428bcf11b383370ef70e3bfcc38" @@ -14461,11 +14480,6 @@ fsevents@^1.2.7: bindings "^1.5.0" nan "^2.12.1" -fsevents@^2.1.2, fsevents@^2.3.2: - version "2.3.2" - resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a" - integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA== - fsevents@~2.3.2: version "2.3.3" resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" @@ -20331,6 +20345,20 @@ pkg-types@^1.1.0: mlly "^1.6.1" pathe "^1.1.2" +playwright-core@1.44.1: + version "1.44.1" + resolved "https://registry.yarnpkg.com/playwright-core/-/playwright-core-1.44.1.tgz#53ec975503b763af6fc1a7aa995f34bc09ff447c" + integrity sha512-wh0JWtYTrhv1+OSsLPgFzGzt67Y7BE/ZS3jEqgGBlp2ppp1ZDj8c+9IARNW4dwf1poq5MgHreEM2KV/GuR4cFA== + +playwright@1.44.1: + version "1.44.1" + resolved "https://registry.yarnpkg.com/playwright/-/playwright-1.44.1.tgz#5634369d777111c1eea9180430b7a184028e7892" + integrity sha512-qr/0UJ5CFAtloI3avF95Y0L1xQo6r3LQArLIg/z/PoGJ6xa+EwzrwO5lpNr/09STxdHuUoP2mvuELJS+hLdtgg== + dependencies: + playwright-core "1.44.1" + optionalDependencies: + fsevents "2.3.2" + plur@^2.1.2: version "2.1.2" resolved "https://registry.yarnpkg.com/plur/-/plur-2.1.2.tgz#7482452c1a0f508e3e344eaec312c91c29dc655a"