Compare commits

..

81 Commits

Author SHA1 Message Date
Szilárd Dóró
fa045eed15 Merge pull request #1808 from nhost/changeset-release/main
chore: update versions
2023-04-03 16:35:43 +02:00
github-actions[bot]
61c0583b6d chore: update versions 2023-04-03 13:56:32 +00:00
Szilárd Dóró
1343a6f252 Merge pull request #1806 from nhost/fix/failed-subscriptions
fix: don't open unnecessary connections
2023-04-03 15:55:27 +02:00
Szilárd Dóró
0d73e87a83 fix: don't open unnecessary connections 2023-04-03 15:14:28 +02:00
Szilárd Dóró
1ee0d332bf Merge pull request #1797 from nhost/changeset-release/main
chore: update versions
2023-04-03 13:50:27 +02:00
github-actions[bot]
130ce49c76 chore: update versions 2023-04-03 10:06:30 +00:00
Szilárd Dóró
6be6d6475a Merge pull request #1804 from nhost/fix/date-input
fix(dashboard): don't break logs page
2023-04-03 11:58:37 +02:00
Szilárd Dóró
177b146b93 Merge pull request #1799 from nhost/renovate/urql-4.x
chore(deps): update dependency urql to v4
2023-04-03 11:39:47 +02:00
Szilárd Dóró
3cb673000a fix: don't break logs page 2023-04-03 11:38:34 +02:00
Szilárd Dóró
09cf5d4b39 chore: add changeset 2023-04-03 10:07:16 +02:00
Szilárd Dóró
48c0061a0b Merge pull request #1772 from wollerman/patch-2
Update serverless-functions to link to event trigger docs
2023-04-03 09:47:01 +02:00
Szilárd Dóró
0795d1c6a1 chore: move new text 2023-04-03 09:34:02 +02:00
renovate[bot]
45a23dd1bf chore(deps): update dependency urql to v4 2023-04-03 07:30:32 +00:00
Szilárd Dóró
bb8e3454df Merge pull request #1790 from nhost/renovate/react-monorepo
chore(deps): update dependency @types/react to v18.0.32
2023-04-03 09:28:01 +02:00
Szilárd Dóró
6a290bb297 chore: add changeset 2023-04-03 09:10:03 +02:00
renovate[bot]
80baec7356 chore(deps): update dependency @types/react to v18.0.32 2023-04-02 09:44:20 +00:00
Szilárd Dóró
8e43297564 Merge pull request #1798 from nhost/feat/project-creator
feat(dashboard): show project creator
2023-03-31 14:49:16 +02:00
Szilárd Dóró
bb8eb9e387 fix: fix assertion in test 2023-03-31 13:55:05 +02:00
Szilárd Dóró
5b0dc6cb19 fix: use optional chaining in overview header 2023-03-31 13:45:58 +02:00
Szilárd Dóró
826112afd0 fix: don't show upgrade button for pro projects 2023-03-31 13:32:16 +02:00
Szilárd Dóró
97105c390d chore: remove test file 2023-03-31 13:26:59 +02:00
Szilárd Dóró
8e3707ff2c Merge branch 'main' into feat/project-creator 2023-03-31 13:25:32 +02:00
Szilárd Dóró
7453bf3b6a chore: add changeset 2023-03-31 13:25:25 +02:00
Szilárd Dóró
bd739383d2 Merge pull request #1796 from nhost/chore/improved-auth-tests
chore(tests): improve auth page tests
2023-03-31 13:19:44 +02:00
Szilárd Dóró
f75e2e41db chore: prefix email addresses 2023-03-31 10:48:54 +02:00
Szilárd Dóró
7328491be0 feat: add relative time to creator info 2023-03-30 20:37:53 +02:00
Szilárd Dóró
11b4d12f12 Merge pull request #1794 from nhost/changeset-release/main
chore: update versions
2023-03-30 19:56:33 +02:00
Szilárd Dóró
6c24d56b1d fix: remove test.only 2023-03-30 17:41:14 +02:00
Szilárd Dóró
0a523f4b45 feat: add project creator to overviews 2023-03-30 17:37:21 +02:00
Szilárd Dóró
12301e6551 fix: use correct @nhost/apollo version 2023-03-30 15:57:43 +02:00
Szilárd Dóró
8440d0389e chore: restructure auth tests 2023-03-30 15:55:58 +02:00
Szilárd Dóró
c166dad0f8 chore: add changeset 2023-03-30 13:49:14 +02:00
Szilárd Dóró
e31d39b3d2 feat: incorporate global setup in projects 2023-03-30 13:44:34 +02:00
Szilárd Dóró
090f9cef86 chore: extend user management tests 2023-03-30 13:35:06 +02:00
github-actions[bot]
74e52cac2d chore: update versions 2023-03-30 09:07:41 +00:00
Szilárd Dóró
f17823760a Merge pull request #1795 from nhost/fix/presigned-urls
fix: don't alter URLs when no transformation parameters are available
2023-03-30 11:06:32 +02:00
Szilárd Dóró
bb8803a1e3 fix: don't alter URLs 2023-03-30 10:41:57 +02:00
Szilárd Dóró
b846291331 Merge pull request #1793 from nhost/fix/export-issue
fix: don't use conflicting names
2023-03-30 10:07:24 +02:00
Szilárd Dóró
2b2fb94f00 chore: add type checking step 2023-03-30 09:42:23 +02:00
Szilárd Dóró
551760c4f0 fix: don't break builds 2023-03-30 09:37:39 +02:00
Szilárd Dóró
5ae5a8e77d fix: don't break builds 2023-03-30 09:31:54 +02:00
Szilárd Dóró
56aae0c964 fix: don't break builds 2023-03-30 09:28:34 +02:00
Szilárd Dóró
a0e093d77b fix: don't use conflicting names 2023-03-30 09:16:30 +02:00
Szilárd Dóró
5e82e1b3da Merge pull request #1784 from nhost/changeset-release/main
chore: update versions
2023-03-29 09:20:48 +02:00
github-actions[bot]
e618b705e7 chore: update versions 2023-03-28 15:52:47 +00:00
Szilárd Dóró
a232c9f0f6 Merge pull request #1789 from nhost/fix/azuread-description
fix(dashboard): use correct description for Azure AD
2023-03-28 17:50:51 +02:00
Szilárd Dóró
bf4644ea10 fix: use correct description for Azure AD 2023-03-28 16:52:54 +02:00
Szilárd Dóró
0aca907ea4 Merge pull request #1788 from nhost/fix/azuread-provider-name
fix: correct typos in Azure AD
2023-03-28 16:25:59 +02:00
Szilárd Dóró
394f4c4174 fix: correct typos in Azure AD 2023-03-28 16:25:26 +02:00
Szilárd Dóró
8fef08a150 Merge pull request #1786 from nhost/renovate/turbo-1.x
chore(deps): update dependency turbo to v1.8.6
2023-03-28 16:16:57 +02:00
Szilárd Dóró
1bd2c37301 chore: bump turbo in the Dockerfile 2023-03-28 15:54:37 +02:00
renovate[bot]
5cdb70bd81 chore(deps): update dependency turbo to v1.8.6 2023-03-28 12:01:36 +00:00
Szilárd Dóró
1a5f80e1b6 Merge pull request #1785 from nhost/renovate/react-monorepo
chore(deps): update dependency @types/react to v18.0.30
2023-03-28 13:59:29 +02:00
Szilárd Dóró
59e0cb00c5 Merge pull request #1787 from nhost/feat/azuread-provider 2023-03-28 12:25:42 +02:00
Szilárd Dóró
406b0f2cb7 Merge pull request #1163 from dipakparmar/feat/dashboard-azuread-settings
feat(dashboard): add azure ad provider settings
2023-03-28 10:52:17 +02:00
Szilárd Dóró
d329b6218f chore: add changeset 2023-03-28 10:46:50 +02:00
Szilárd Dóró
335b58670e Merge branch 'renovate/react-monorepo' of https://github.com/nhost/nhost into renovate/react-monorepo 2023-03-28 10:43:08 +02:00
renovate[bot]
efa2d89067 chore(deps): update dependency @types/react to v18.0.30 2023-03-28 08:35:55 +00:00
Szilárd Dóró
77ce4bd738 Merge pull request #1783 from nhost/fix/random-words
fix(tests): avoid name collision in database tests
2023-03-28 10:33:33 +02:00
Szilárd Dóró
017adea700 chore: update comment 2023-03-28 10:04:38 +02:00
Dipak Parmar
378284faa8 chore(dashboard): remove docs and title for now from azuread component
Signed-off-by: Dipak Parmar <hi@dipak.tech>
2023-03-27 23:44:40 -07:00
renovate[bot]
e5e2d114b1 chore(deps): update dependency @types/react to v18.0.30 2023-03-27 19:03:37 +00:00
Szilárd Dóró
5e3dbdeb7d Merge pull request #1781 from nhost/renovate/react-monorepo
chore(deps): update dependency @types/react to v18.0.29
2023-03-27 20:55:47 +02:00
Szilárd Dóró
98b777491a fix: improve flaky tests 2023-03-27 18:13:10 +02:00
Szilárd Dóró
71de870cb0 fix: use admin secret as env var 2023-03-27 17:29:09 +02:00
Szilárd Dóró
74d4deba28 feat: cleanup public schema after tests 2023-03-27 17:00:07 +02:00
Szilárd Dóró
cb248f0d30 chore: add changeset 2023-03-27 15:44:08 +02:00
Szilárd Dóró
09e4f1eb34 fix: avoid duplicate table names in tests 2023-03-27 15:16:40 +02:00
Szilárd Dóró
6e1f03eaee chore: accomodate changes to API 2023-03-27 11:57:24 +02:00
Szilárd Dóró
867c807699 chore: add changeset 2023-03-27 11:21:42 +02:00
renovate[bot]
d0673d7825 chore(deps): update dependency @types/react to v18.0.29 2023-03-27 07:50:19 +00:00
Dipak Parmar
106f23dcfa fixdashboard-settings): remove extra whitespace azuread provider import in settings
Signed-off-by: Dipak Parmar <hi@dipak.tech>
2023-03-27 00:48:56 -07:00
Dipak Parmar
83ef755822 feat(dashboard-settings): update azuread provider settings component
Signed-off-by: Dipak Parmar <hi@dipak.tech>
2023-03-27 00:47:09 -07:00
Dipak Parmar
b7703ffd70 feat(graphql): add azuread to signinmethods query
Signed-off-by: Dipak Parmar <hi@dipak.tech>
2023-03-27 00:46:30 -07:00
Dipak Parmar
340ea5b115 chore: merge branch 'main' into feat/dashboard-azuread-settings
* main: (1322 commits)
  chore(ci): adjust preview fetcher
  chore: add changeset
  fix: fetch valid previews only
  fix: use correct Vercel token
  fix: use staging project ID
  chore: use dynamic test URL
  fix(deps): update docusaurus monorepo to v2.4.0
  chore(hasura-storage-js): improve presignedUrl test
  fix: remove test.only call
  chore: add tests for table deletion
  chore: update versions
  fix: potential subscription fix
  Fix import in docs
  fix: remove `test.only` call
  chore: add remaining table creation tests
  chore: add foreign key constraint test
  chore: add extra database UI tests
  chore: restructure tests, add basic table creation test
  chore: update versions
  chore: add changeset
  ...

Signed-off-by: Dipak Parmar <hi@dipak.tech>
2023-03-26 19:16:40 -07:00
Matt Wollerman
776eca3fb5 Update serverless-functions to link to event trigger docs 2023-03-23 09:06:20 -04:00
Dipak Parmar
ce4b655c55 fix: correct typos 2022-11-22 19:47:21 -08:00
Dipak Parmar
dc57d31ec9 fix: correct extra space in azureadprovidersettings dir 2022-11-22 19:45:38 -08:00
Dipak Parmar
ea29fd6b73 feat(dashboard-settings): add azuread provider to settings 2022-11-21 20:30:53 -08:00
Dipak Parmar
d8e4073957 feat(dashboard-settings): add azuread provider settings component 2022-11-21 20:29:34 -08:00
Dipak Parmar
3f399a54a3 feat(graphql): add azuread to signinmethods query 2022-11-21 20:28:50 -08:00
114 changed files with 2005 additions and 712 deletions

View File

@@ -24,6 +24,7 @@ env:
NHOST_TEST_PROJECT_NAME: ${{ vars.NHOST_TEST_PROJECT_NAME }}
NHOST_TEST_USER_EMAIL: ${{ secrets.NHOST_TEST_USER_EMAIL }}
NHOST_TEST_USER_PASSWORD: ${{ secrets.NHOST_TEST_USER_PASSWORD }}
NHOST_TEST_PROJECT_ADMIN_SECRET: ${{ secrets.NHOST_TEST_PROJECT_ADMIN_SECRET }}
jobs:
build:

View File

@@ -21,6 +21,7 @@ module.exports = {
'error',
{ allowArrowFunctions: true, allowFunctions: true },
],
'import/prefer-default-export': 'off',
'import/no-extraneous-dependencies': ['error', { devDependencies: true }],
curly: ['error', 'all'],
'no-restricted-exports': 'off',

View File

@@ -53,4 +53,5 @@ tailwind.json
/test-results/
/playwright-report/
/playwright/.cache/
storageState.json
storageState.json
e2e/.auth/*

View File

@@ -1,5 +1,40 @@
# @nhost/dashboard
## 0.14.3
### Patch Changes
- @nhost/react-apollo@5.0.16
## 0.14.2
### Patch Changes
- 3cb67300: fix(logs): don't break UI when clearing time picker
- 7453bf3b: feat(projects): show project creator info
- c166dad0: chore(tests): improve auth page tests
- 6a290bb2: chore(deps): bump `@types/react` to 18.0.32
## 0.14.1
### Patch Changes
- @nhost/react-apollo@5.0.15
- @nhost/nextjs@1.13.19
## 0.14.0
### Minor Changes
- 6e1f03ea: feat(dashboard): add support for the Azure AD provider
### Patch Changes
- 1bd2c373: chore(deps): bump `turbo` to 1.8.6
- d329b621: chore(deps): bump `@types/react` to 18.0.30
- cb248f0d: fix(tests): avoid name collision in database tests
- 867c8076: chore(deps): bump `@types/react` to 18.0.29
## 0.13.10
### Patch Changes

View File

@@ -3,7 +3,7 @@ RUN apk add --no-cache libc6-compat
RUN apk update
WORKDIR /app
RUN yarn global add turbo@1.8.3
RUN yarn global add turbo@1.8.6
COPY . .
RUN turbo prune --scope="@nhost/dashboard" --docker

View File

@@ -128,4 +128,5 @@ NHOST_TEST_USER_EMAIL=<test_user_email>
NHOST_TEST_USER_PASSWORD=<test_user_password>
NHOST_TEST_WORKSPACE_NAME=<test_workspace_name>
NHOST_TEST_PROJECT_NAME=<test_project_name>
NHOST_TEST_PROJECT_ADMIN_SECRET=<test_project_admin_secret>
```

View File

@@ -0,0 +1,50 @@
import {
TEST_PROJECT_NAME,
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { createUser, generateTestEmail, openProject } from '@/e2e/utils';
import { faker } from '@faker-js/faker';
import test, { expect } from '@playwright/test';
test('should be able to ban and unban a user', async ({ page }) => {
await page.goto('/');
await openProject({
page,
projectName: TEST_PROJECT_NAME,
workspaceSlug: TEST_WORKSPACE_SLUG,
projectSlug: TEST_PROJECT_SLUG,
});
await page
.getByRole('navigation', { name: /main navigation/i })
.getByRole('link', { name: /auth/i })
.click();
await page.waitForURL(`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/users`);
const email = generateTestEmail();
const password = faker.internet.password();
await createUser({ page, email, password });
await page
.getByRole('button', { name: `View ${email}`, exact: true })
.click();
await page.getByRole('button', { name: /actions/i }).click();
await page.getByRole('menuitem', { name: /ban user/i }).click();
await expect(
page.getByText(/user has been banned successfully./i),
).toBeVisible();
await expect(page.locator('form').getByText(/^banned$/i)).toBeVisible();
await page.getByRole('button', { name: /actions/i }).click();
await page.getByRole('menuitem', { name: /unban user/i }).click();
await expect(
page.getByText(/user has been unbanned successfully./i),
).toBeVisible();
await expect(page.locator('form').getByText(/^banned$/i)).not.toBeVisible();
});

View File

@@ -0,0 +1,65 @@
import {
TEST_PROJECT_NAME,
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { createUser, generateTestEmail, openProject } from '@/e2e/utils';
import { faker } from '@faker-js/faker';
import type { Page } from '@playwright/test';
import test, { expect } from '@playwright/test';
let page: Page;
test.beforeAll(async ({ browser }) => {
page = await browser.newPage();
});
test.beforeEach(async () => {
await page.goto('/');
await openProject({
page,
projectName: TEST_PROJECT_NAME,
workspaceSlug: TEST_WORKSPACE_SLUG,
projectSlug: TEST_PROJECT_SLUG,
});
await page
.getByRole('navigation', { name: /main navigation/i })
.getByRole('link', { name: /auth/i })
.click();
await page.waitForURL(`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/users`);
});
test.afterAll(async () => {
await page.close();
});
test('should create a user', async () => {
const email = generateTestEmail();
const password = faker.internet.password();
await createUser({ page, email, password });
await expect(
page.getByRole('button', { name: `View ${email}`, exact: true }),
).toBeVisible();
});
test('should not be able to create a user with an existing email', async () => {
const email = generateTestEmail();
const password = faker.internet.password();
await createUser({ page, email, password });
await expect(
page.getByRole('button', { name: `View ${email}`, exact: true }),
).toBeVisible();
await createUser({ page, email, password });
await expect(
page.getByRole('dialog').getByText(/email already in use/i),
).toBeVisible();
});

View File

@@ -0,0 +1,96 @@
import {
TEST_PROJECT_NAME,
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { createUser, generateTestEmail, openProject } from '@/e2e/utils';
import { faker } from '@faker-js/faker';
import type { Page } from '@playwright/test';
import test, { expect } from '@playwright/test';
let page: Page;
test.beforeAll(async ({ browser }) => {
page = await browser.newPage();
});
test.beforeEach(async () => {
await page.goto('/');
await openProject({
page,
projectName: TEST_PROJECT_NAME,
workspaceSlug: TEST_WORKSPACE_SLUG,
projectSlug: TEST_PROJECT_SLUG,
});
await page
.getByRole('navigation', { name: /main navigation/i })
.getByRole('link', { name: /auth/i })
.click();
await page.waitForURL(`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/users`);
});
test.afterAll(async () => {
await page.close();
});
test('should be able to delete a user', async () => {
const email = generateTestEmail();
const password = faker.internet.password();
await createUser({ page, email, password });
await expect(
page.getByRole('button', { name: `View ${email}`, exact: true }),
).toBeVisible();
await page
.getByRole('button', { name: `More options for ${email}`, exact: true })
.click();
await page.getByRole('menuitem', { name: /delete user/i }).click();
await expect(page.getByRole('dialog')).toBeVisible();
await expect(
page.getByRole('heading', { name: /delete user/i }),
).toBeVisible();
await expect(
page.getByText(`Are you sure you want to delete the "${email}" user?`),
).toBeVisible();
await page.getByRole('button', { name: /delete/i, exact: true }).click();
await expect(page.getByRole('dialog')).not.toBeVisible();
await expect(
page.getByRole('button', { name: `View ${email}`, exact: true }),
).not.toBeVisible();
});
test('should be able to delete a user from the details page', async () => {
const email = generateTestEmail();
const password = faker.internet.password();
await createUser({ page, email, password });
await page
.getByRole('button', { name: `View ${email}`, exact: true })
.click();
await page.getByRole('button', { name: /actions/i }).click();
await page.getByRole('menuitem', { name: /delete user/i }).click();
await expect(page.getByRole('dialog')).toBeVisible();
await expect(
page.getByRole('heading', { name: /delete user/i }),
).toBeVisible();
await expect(
page.getByText(`Are you sure you want to delete the "${email}" user?`),
).toBeVisible();
await page.getByRole('button', { name: /delete/i, exact: true }).click();
await expect(
page.getByRole('button', { name: `View ${email}`, exact: true }),
).not.toBeVisible();
});

View File

@@ -1,93 +0,0 @@
import {
TEST_PROJECT_NAME,
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { openProject } from '@/e2e/utils';
import type { Page } from '@playwright/test';
import { expect, test } from '@playwright/test';
let page: Page;
test.describe.configure({ mode: 'serial' });
test.beforeAll(async ({ browser }) => {
page = await browser.newPage();
await page.goto('/');
await openProject({
page,
projectName: TEST_PROJECT_NAME,
workspaceSlug: TEST_WORKSPACE_SLUG,
projectSlug: TEST_PROJECT_SLUG,
});
await page
.getByRole('navigation', { name: /main navigation/i })
.getByRole('link', { name: /auth/i })
.click();
await page.waitForURL(`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/users`);
});
test.afterAll(async () => {
await page.close();
});
test('should create a user', async () => {
await expect(
page.getByRole('heading', { name: /there are no users yet/i }),
).toBeVisible();
await page
.getByRole('button', { name: /create user/i })
.first()
.click();
await expect(page.getByRole('dialog')).toBeVisible();
await expect(
page.getByRole('heading', { name: /create user/i }),
).toBeVisible();
await page
.getByRole('textbox', { name: /email/i })
.fill('testuser@example.com');
await page.getByRole('textbox', { name: /password/i }).fill('test.password');
await page.getByRole('button', { name: /create/i, exact: true }).click();
await expect(page.getByRole('dialog')).not.toBeVisible();
await expect(
page.getByRole('button', { name: /view testuser@example.com/i }),
).toBeVisible();
});
test('should delete a user', async () => {
await expect(
page.getByRole('button', { name: /view testuser@example.com/i }),
).toBeVisible();
await page
.getByRole('button', { name: /more options for testuser@example.com/i })
.click();
await page.getByRole('menuitem', { name: /delete user/i }).click();
await expect(page.getByRole('dialog')).toBeVisible();
await expect(
page.getByRole('heading', { name: /delete user/i }),
).toBeVisible();
await expect(
page.getByText(
/are you sure you want to delete the "testuser@example.com" user?/i,
),
).toBeVisible();
await page.getByRole('button', { name: /delete/i, exact: true }).click();
await expect(page.getByRole('dialog')).not.toBeVisible();
await expect(
page.getByRole('heading', { name: /there are no users yet/i }),
).toBeVisible();
});

View File

@@ -0,0 +1,103 @@
import {
TEST_PROJECT_NAME,
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { createUser, generateTestEmail, openProject } from '@/e2e/utils';
import { faker } from '@faker-js/faker';
import type { Page } from '@playwright/test';
import { expect, test } from '@playwright/test';
let page: Page;
test.beforeAll(async ({ browser }) => {
page = await browser.newPage();
});
test.beforeEach(async () => {
await page.goto('/');
await openProject({
page,
projectName: TEST_PROJECT_NAME,
workspaceSlug: TEST_WORKSPACE_SLUG,
projectSlug: TEST_PROJECT_SLUG,
});
await page
.getByRole('navigation', { name: /main navigation/i })
.getByRole('link', { name: /auth/i })
.click();
await page.waitForURL(`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/users`);
});
test.afterAll(async () => {
await page.close();
});
test('should be able to verify the email of a user', async () => {
const email = generateTestEmail();
const password = faker.internet.password();
await createUser({ page, email, password });
await page
.getByRole('button', { name: `View ${email}`, exact: true })
.click();
await expect(
page.getByRole('checkbox', { name: /email verified/i }),
).not.toBeChecked();
await page.getByRole('checkbox', { name: /email verified/i }).check();
await page.getByRole('button', { name: /save/i }).click();
await expect(
page.getByText(/user settings have been updated successfully./i),
).toBeVisible();
await page
.getByRole('button', { name: `View ${email}`, exact: true })
.click();
await expect(
page.getByRole('checkbox', { name: /email verified/i }),
).toBeChecked();
});
test('should be able to verify the phone number of a user', async () => {
const email = generateTestEmail();
const password = faker.internet.password();
const phoneNumber = faker.phone.number();
await createUser({ page, email, password });
await page
.getByRole('button', { name: `View ${email}`, exact: true })
.click();
await expect(
page.getByRole('checkbox', { name: /phone number verified/i }),
).toBeDisabled();
await page.getByRole('textbox', { name: /phone number/i }).fill(phoneNumber);
await page.getByRole('checkbox', { name: /phone number verified/i }).check();
await page.getByRole('button', { name: /save/i }).click();
await expect(
page.getByText(/user settings have been updated successfully./i),
).toBeVisible();
await page
.getByRole('button', { name: `View ${email}`, exact: true })
.click();
await expect(
page.getByRole('textbox', { name: /phone number/i }),
).toHaveValue(phoneNumber);
await expect(
page.getByRole('checkbox', { name: /phone number verified/i }),
).toBeChecked();
});

View File

@@ -7,11 +7,15 @@ import { openProject, prepareTable } from '@/e2e/utils';
import { faker } from '@faker-js/faker';
import type { Page } from '@playwright/test';
import { expect, test } from '@playwright/test';
import { snakeCase } from 'snake-case';
let page: Page;
test.beforeAll(async ({ browser }) => {
page = await browser.newPage();
});
test.beforeEach(async () => {
await page.goto('/');
await openProject({
@@ -35,7 +39,7 @@ test('should create a simple table', async () => {
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const tableName = faker.random.word().toLowerCase();
const tableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -63,7 +67,7 @@ test('should create a table with unique constraints', async () => {
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const tableName = faker.random.word().toLowerCase();
const tableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -92,7 +96,7 @@ test('should create a table with nullable columns', async () => {
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const tableName = faker.random.word().toLowerCase();
const tableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -121,7 +125,7 @@ test('should create a table with an identity column', async () => {
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const tableName = faker.random.word().toLowerCase();
const tableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -153,7 +157,7 @@ test('should create table with foreign key constraint', async () => {
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const firstTableName = faker.random.word().toLowerCase();
const firstTableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -175,7 +179,7 @@ test('should create table with foreign key constraint', async () => {
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const secondTableName = faker.random.word().toLowerCase();
const secondTableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -234,7 +238,7 @@ test('should not be able to create a table with a name that already exists', asy
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const tableName = faker.random.word().toLowerCase();
const tableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,

View File

@@ -3,15 +3,19 @@ import {
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { openProject, prepareTable } from '@/e2e/utils';
import { deleteTable, openProject, prepareTable } from '@/e2e/utils';
import { faker } from '@faker-js/faker';
import type { Page } from '@playwright/test';
import { expect, test } from '@playwright/test';
import { snakeCase } from 'snake-case';
let page: Page;
test.beforeAll(async ({ browser }) => {
page = await browser.newPage();
});
test.beforeEach(async () => {
await page.goto('/');
await openProject({
@@ -32,7 +36,7 @@ test.afterAll(async () => {
});
test('should delete a table', async () => {
const tableName = faker.random.word().toLowerCase();
const tableName = snakeCase(faker.lorem.words(3));
await page.getByRole('button', { name: /new table/i }).click();
@@ -52,26 +56,11 @@ test('should delete a table', async () => {
`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/database/browser/default/public/${tableName}`,
);
const tableLink = page.getByRole('link', {
await deleteTable({
page,
name: tableName,
exact: true,
});
await tableLink.hover();
await page
.getByRole('listitem')
.filter({ hasText: tableName })
.getByRole('button')
.click();
await page.getByRole('menuitem', { name: /delete table/i }).click();
await expect(
page.getByRole('heading', { name: /delete table/i }),
).toBeVisible();
await page.getByRole('button', { name: /delete/i }).click();
// navigate to next URL
await page.waitForURL(
`/${TEST_WORKSPACE_SLUG}/${TEST_PROJECT_SLUG}/database/browser/default/public/**`,
@@ -86,7 +75,7 @@ test('should not be able to delete a table if other tables have foreign keys ref
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const firstTableName = faker.random.word().toLowerCase();
const firstTableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -108,7 +97,7 @@ test('should not be able to delete a table if other tables have foreign keys ref
await page.getByRole('button', { name: /new table/i }).click();
await expect(page.getByText(/create a new table/i)).toBeVisible();
const secondTableName = faker.random.word().toLowerCase();
const secondTableName = snakeCase(faker.lorem.words(3));
await prepareTable({
page,
@@ -163,26 +152,11 @@ test('should not be able to delete a table if other tables have foreign keys ref
).toBeVisible();
// try to delete the first table that is referenced by the second table
const tableLink = page.getByRole('link', {
await deleteTable({
page,
name: firstTableName,
exact: true,
});
await tableLink.hover();
await page
.getByRole('listitem')
.filter({ hasText: firstTableName })
.getByRole('button')
.click();
await page.getByRole('menuitem', { name: /delete table/i }).click();
await expect(
page.getByRole('heading', { name: /delete table/i }),
).toBeVisible();
await page.getByRole('button', { name: /delete/i }).click();
await expect(
page.getByText(
/constraint [a-zA-Z_]+ on table [a-zA-Z_]+ depends on table [a-zA-Z_]+/i,

View File

@@ -31,6 +31,12 @@ export const TEST_PROJECT_SLUG = slugify(TEST_PROJECT_NAME, {
strict: true,
});
/**
* Hasura admin secret of the test project to use.
*/
export const TEST_PROJECT_ADMIN_SECRET =
process.env.NHOST_TEST_PROJECT_ADMIN_SECRET;
/**
* Email of the test account to use.
*/

View File

@@ -72,7 +72,7 @@ test("should show the project's name, the Upgrade button and the Settings button
await expect(
page.getByRole('heading', { name: TEST_PROJECT_NAME }),
).toBeVisible();
await expect(page.getByText(/free plan/i)).toBeVisible();
await expect(page.getByText(/starter/i)).toBeVisible();
await expect(page.getByRole('button', { name: /upgrade/i })).toBeVisible();
await expect(
page.getByRole('main').getByRole('link', { name: /settings/i }),

View File

@@ -0,0 +1,20 @@
import {
TEST_DASHBOARD_URL,
TEST_USER_EMAIL,
TEST_USER_PASSWORD,
} from '@/e2e/env';
import { test as setup } from '@playwright/test';
setup('authenticate user', async ({ page }) => {
await page.goto('/');
await page.waitForURL('/signin');
await page.getByRole('link', { name: /continue with email/i }).click();
await page.waitForURL('/signin/email');
await page.getByLabel('Email').fill(TEST_USER_EMAIL);
await page.getByLabel('Password').fill(TEST_USER_PASSWORD);
await page.getByRole('button', { name: /sign in/i }).click();
await page.waitForURL(TEST_DASHBOARD_URL);
await page.context().storageState({ path: 'e2e/.auth/user.json' });
});

View File

@@ -1,3 +1,4 @@
import { faker } from '@faker-js/faker';
import type { Page } from '@playwright/test';
/**
@@ -66,18 +67,25 @@ export async function prepareTable({
// set type
await page
.getByRole('table')
.getByRole('combobox', { name: /type/i })
.nth(index)
.fill(type);
await page.getByRole('option', { name: type }).first().click();
.type(type);
await page
.getByRole('table')
.getByRole('option', { name: type })
.first()
.click();
// optionally set default value
if (defaultValue) {
await page
.getByRole('table')
.getByRole('combobox', { name: /default value/i })
.first()
.fill(defaultValue);
.nth(index)
.type(defaultValue);
await page
.getByRole('table')
.getByRole('option', { name: defaultValue })
.first()
.click();
@@ -111,3 +119,71 @@ export async function prepareTable({
await page.getByRole('button', { name: /primary key/i }).click();
await page.getByRole('option', { name: primaryKey, exact: true }).click();
}
/**
* Deletes a table with the given name.
*
* @param page - The Playwright page object.
* @param name - The name of the table to delete.
* @returns A promise that resolves when the table is deleted.
*/
export async function deleteTable({
page,
name,
}: {
page: Page;
name: string;
}) {
const tableLink = page.getByRole('link', {
name,
exact: true,
});
await tableLink.hover();
await page
.getByRole('listitem')
.filter({ hasText: name })
.getByRole('button')
.click();
await page.getByRole('menuitem', { name: /delete table/i }).click();
await page.getByRole('button', { name: /delete/i }).click();
}
/**
* Creates a new user.
*
* @param page - The Playwright page object.
* @param email - The email of the user to create.
* @param password - The password of the user to create.
* @returns A promise that resolves when the user is created.
*/
export async function createUser({
page,
email,
password,
}: {
page: Page;
email: string;
password: string;
}) {
await page
.getByRole('button', { name: /create user/i })
.first()
.click();
await page.getByRole('textbox', { name: /email/i }).fill(email);
await page.getByRole('textbox', { name: /password/i }).fill(password);
await page.getByRole('button', { name: /create/i, exact: true }).click();
}
/**
* Generates a test email address with the given prefix (if provided).
*
* @param prefix - The prefix to use for the email address. (Default: `Nhost_Test_`)
*/
export function generateTestEmail(prefix: string = 'Nhost_Test_') {
const email = faker.internet.email();
return [prefix, email].join('');
}

View File

@@ -1,27 +0,0 @@
import { chromium } from '@playwright/test';
import {
TEST_DASHBOARD_URL,
TEST_USER_EMAIL,
TEST_USER_PASSWORD,
} from './e2e/env';
async function globalSetup() {
const browser = await chromium.launch();
const page = await browser.newPage();
await page.goto(TEST_DASHBOARD_URL);
await page.waitForURL(`${TEST_DASHBOARD_URL}/signin`);
await page.getByRole('link', { name: /continue with email/i }).click();
await page.waitForURL(`${TEST_DASHBOARD_URL}/signin/email`);
await page.getByLabel('Email').fill(TEST_USER_EMAIL);
await page.getByLabel('Password').fill(TEST_USER_PASSWORD);
await page.getByRole('button', { name: /sign in/i }).click();
await page.waitForURL(TEST_DASHBOARD_URL);
await page.context().storageState({ path: 'storageState.json' });
await browser.close();
}
export default globalSetup;

View File

@@ -0,0 +1,66 @@
import {
TEST_DASHBOARD_URL,
TEST_PROJECT_ADMIN_SECRET,
TEST_PROJECT_NAME,
TEST_PROJECT_SLUG,
TEST_WORKSPACE_SLUG,
} from '@/e2e/env';
import { openProject } from '@/e2e/utils';
import { chromium } from '@playwright/test';
async function globalTeardown() {
const browser = await chromium.launch();
const context = await browser.newContext({
baseURL: TEST_DASHBOARD_URL,
storageState: 'e2e/.auth/user.json',
});
const page = await context.newPage();
await page.goto('/');
await openProject({
page,
projectName: TEST_PROJECT_NAME,
workspaceSlug: TEST_WORKSPACE_SLUG,
projectSlug: TEST_PROJECT_SLUG,
});
const pagePromise = context.waitForEvent('page');
await page.getByRole('link', { name: /hasura/i }).click();
await page.getByRole('link', { name: /open hasura/i }).click();
const hasuraPage = await pagePromise;
await hasuraPage.waitForLoadState();
const adminSecretInput = hasuraPage.getByPlaceholder(/enter admin-secret/i);
// note: a more ideal way would be to paste from clipboard, but Playwright
// doesn't support that yet
await adminSecretInput.fill(TEST_PROJECT_ADMIN_SECRET);
await adminSecretInput.press('Enter');
// note: getByRole doesn't work here
await hasuraPage.locator('a', { hasText: /data/i }).click();
await hasuraPage.getByRole('link', { name: /sql/i }).click();
await hasuraPage.getByRole('textbox').fill(`
DO $$ DECLARE
tablename text;
BEGIN
FOR tablename IN
SELECT table_name FROM information_schema.tables
WHERE table_schema = 'public'
LOOP
EXECUTE 'DROP TABLE IF EXISTS public.' || quote_ident(tablename) || ' CASCADE';
END LOOP;
END $$;
`);
await hasuraPage.getByRole('button', { name: /run!/i }).click();
await hasuraPage.getByText(/sql executed!/i).waitFor();
}
export default globalTeardown;

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/dashboard",
"version": "0.13.10",
"version": "0.14.3",
"private": true,
"scripts": {
"preinstall": "npx only-allow pnpm",
@@ -106,7 +106,7 @@
"@types/lodash.debounce": "^4.0.7",
"@types/node": "^16.11.7",
"@types/pluralize": "^0.0.29",
"@types/react": "18.0.28",
"@types/react": "18.0.32",
"@types/react-dom": "18.0.11",
"@types/react-table": "^7.7.12",
"@types/testing-library__jest-dom": "^5.14.5",
@@ -141,6 +141,7 @@
"prettier-plugin-tailwindcss": "^0.2.0",
"react-date-fns-hooks": "^0.9.4",
"require-from-string": "^2.0.2",
"snake-case": "^3.0.4",
"storybook-addon-next-router": "^4.0.1",
"tailwindcss": "^3.1.2",
"ts-node": "^10.9.1",

View File

@@ -1,5 +1,4 @@
import { defineConfig, devices } from '@playwright/test';
import dotenv from 'dotenv';
import path from 'path';
@@ -16,17 +15,24 @@ export default defineConfig({
retries: process.env.CI ? 2 : 0,
workers: process.env.CI ? 1 : undefined,
reporter: 'html',
globalSetup: require.resolve('./global-setup'),
globalTeardown: require.resolve('./global-teardown'),
use: {
actionTimeout: 0,
trace: 'on-first-retry',
storageState: 'storageState.json',
baseURL: process.env.NHOST_TEST_DASHBOARD_URL,
},
projects: [
{
name: 'setup',
testMatch: ['**/setup/*.setup.ts'],
},
{
name: 'chromium',
use: { ...devices['Desktop Chrome'] },
use: {
...devices['Desktop Chrome'],
storageState: 'e2e/.auth/user.json',
},
dependencies: ['setup'],
},
],
});

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 16 KiB

View File

@@ -9,6 +9,7 @@ import { useGetAllUserWorkspacesAndApplications } from '@/hooks/useGetAllUserWor
import { useNavigationVisible } from '@/hooks/useNavigationVisible';
import useNotFoundRedirect from '@/hooks/useNotFoundRedirect';
import { useSetAppWorkspaceContextFromUserContext } from '@/hooks/useSetAppWorkspaceContextFromUserContext';
import { useCurrentWorkspaceAndProject } from '@/hooks/v2/useCurrentWorkspaceAndProject';
import type { BoxProps } from '@/ui/v2/Box';
import Box from '@/ui/v2/Box';
import { NextSeo } from 'next-seo';
@@ -30,9 +31,15 @@ function ProjectLayoutContent({
...mainContainerProps
} = {},
}: ProjectLayoutProps) {
const { currentApplication, currentWorkspace } =
// TODO: This will be removed once we migrated every occurrence to
// useCurrentWorkspaceAndProject()
const { currentWorkspace, currentApplication } =
useCurrentWorkspaceAndApplication();
const { currentProject, loading, error } = useCurrentWorkspaceAndProject({
fetchPolicy: 'network-only',
});
const router = useRouter();
const shouldDisplayNav = useNavigationVisible();
const isPlatform = useIsPlatform();
@@ -63,10 +70,14 @@ function ProjectLayoutContent({
}
}, [isPlatform, isRestrictedPath, router]);
if (!currentWorkspace || !currentApplication || isRestrictedPath) {
if (!currentWorkspace || !currentApplication || isRestrictedPath || loading) {
return <LoadingScreen />;
}
if (error) {
throw error;
}
if (!isPlatform) {
return (
<>
@@ -104,7 +115,7 @@ function ProjectLayoutContent({
>
{children}
<NextSeo title={currentApplication.name} />
<NextSeo title={currentProject.name} />
</Box>
</>
);

View File

@@ -4,6 +4,7 @@ import { useDropdown } from '@/ui/v2/Dropdown';
import type { InputProps } from '@/ui/v2/Input';
import Input from '@/ui/v2/Input';
import { format, set } from 'date-fns';
import type { ChangeEvent } from 'react';
export interface LogTimePickerProps extends InputProps {
/**
@@ -22,21 +23,21 @@ function LogsTimePicker({
}: any) {
const { handleClose } = useDropdown();
const handleCancel = () => {
function handleCancel() {
handleClose();
};
}
const handleApply = () => {
function handleApply() {
onChange(selectedDate);
handleClose();
};
}
const handleTimePicking = (event) => {
const [hours, minutes, seconds] = event.target.value.split(':');
function handleChange(event: ChangeEvent<HTMLInputElement>) {
const [hours, minutes, seconds] = event.target.value?.split(':') || [];
const hoursNumber = parseInt(hours, 10);
const minutesNumber = parseInt(minutes, 10);
const secondsNumber = parseInt(seconds, 10);
const hoursNumber = parseInt(hours || '0', 10);
const minutesNumber = parseInt(minutes || '0', 10);
const secondsNumber = parseInt(seconds || '0', 10);
const newDate = set(new Date(selectedDate), {
hours: hoursNumber,
@@ -51,7 +52,7 @@ function LogsTimePicker({
}
setSelectedDate(newDate);
};
}
return (
<div className="mx-auto grid grid-flow-row items-center self-center">
@@ -64,7 +65,7 @@ function LogsTimePicker({
formControl: { className: 'grid grid-flow-col gap-x-3' },
label: { sx: { fontSize: '14px' } },
}}
onChange={handleTimePicking}
onChange={handleChange}
type="time"
label="Select Time"
sx={{

View File

@@ -1,11 +1,11 @@
import InfoCard from '@/components/overview/InfoCard';
import { useCurrentWorkspaceAndApplication } from '@/hooks/useCurrentWorkspaceAndApplication';
import { useCurrentWorkspaceAndProject } from '@/hooks/v2/useCurrentWorkspaceAndProject';
import Text from '@/ui/v2/Text';
import Image from 'next/image';
export default function OverviewProjectInfo() {
const { currentApplication } = useCurrentWorkspaceAndApplication();
const { region, subdomain } = currentApplication || {};
const { currentProject } = useCurrentWorkspaceAndProject();
const { region, subdomain } = currentProject || {};
const isRegionAvailable =
region?.awsName && region?.countryCode && region?.city;
@@ -13,14 +13,14 @@ export default function OverviewProjectInfo() {
<div className="grid grid-flow-row content-start gap-6">
<Text variant="h3">Project Info</Text>
{currentApplication && (
{currentProject && (
<div className="grid grid-flow-row gap-3">
<InfoCard
title="Region"
value={region?.awsName}
customValue={
region.countryCode &&
region.city && (
region?.countryCode &&
region?.city && (
<div className="grid grid-flow-col items-center gap-1 self-center">
<Image
src={`/assets/flags/${region.countryCode}.svg`}
@@ -29,7 +29,7 @@ export default function OverviewProjectInfo() {
height={12}
/>
<Text className="text-sm font-medium truncate">
<Text className="truncate text-sm font-medium">
{region.city} ({region.awsName})
</Text>
</div>

View File

@@ -2,20 +2,19 @@ import { ChangePlanModal } from '@/components/applications/ChangePlanModal';
import { useDialog } from '@/components/common/DialogProvider';
import { useUI } from '@/context/UIContext';
import useIsPlatform from '@/hooks/common/useIsPlatform';
import { useCurrentWorkspaceAndApplication } from '@/hooks/useCurrentWorkspaceAndApplication';
import Box from '@/ui/v2/Box';
import { useCurrentWorkspaceAndProject } from '@/hooks/v2/useCurrentWorkspaceAndProject';
import Button from '@/ui/v2/Button';
import Chip from '@/ui/v2/Chip';
import CogIcon from '@/ui/v2/icons/CogIcon';
import Text from '@/ui/v2/Text';
import { formatDistanceToNowStrict, parseISO } from 'date-fns';
import Image from 'next/image';
import Link from 'next/link';
export default function OverviewTopBar() {
const isPlatform = useIsPlatform();
const { currentWorkspace, currentApplication } =
useCurrentWorkspaceAndApplication();
const isPro = !currentApplication?.plan?.isFree;
const { currentWorkspace, currentProject } = useCurrentWorkspaceAndProject();
const isPro = !currentProject?.plan?.isFree;
const { openAlertDialog } = useDialog();
const { maintenanceActive } = useUI();
@@ -44,63 +43,92 @@ export default function OverviewTopBar() {
return (
<div className="grid items-center gap-4 pb-5 md:grid-flow-col md:place-content-between md:py-5">
<div className="grid items-center gap-2 md:grid-flow-col">
<div className="grid items-center gap-4 md:grid-flow-col">
<div className="grid grid-flow-col items-center justify-start gap-2">
<div className="h-10 w-10 overflow-hidden rounded-lg">
<Image
src="/logos/new.svg"
alt="Nhost Logo"
width={40}
height={40}
width={56}
height={56}
/>
</div>
<Text variant="h2" component="h1">
{currentApplication.name}
</Text>
</div>
<Box className="grid grid-flow-col items-center justify-start gap-2">
{isPro ? (
<Chip
className="self-center font-medium"
size="small"
label="Pro Plan"
color="primary"
/>
) : (
<>
<Chip
className="self-center font-medium"
size="small"
label="Free Plan"
color="default"
variant="filled"
/>
<Button
variant="borderless"
className="mr-2"
onClick={() => {
openAlertDialog({
title: 'Upgrade your plan.',
payload: <ChangePlanModal />,
props: {
PaperProps: { className: 'p-0 max-w-xl w-full' },
hidePrimaryAction: true,
hideSecondaryAction: true,
hideTitle: true,
},
});
}}
<div className="grid grid-flow-row">
<div className="grid grid-flow-row items-center justify-start md:grid-flow-col md:gap-3">
<Text
variant="h2"
component="h1"
className="grid grid-flow-col items-center gap-3"
>
Upgrade
</Button>
</>
)}
</Box>
{currentProject.name}
</Text>
{currentProject.creator && (
<Text
color="secondary"
variant="subtitle2"
className="md:hidden"
>
Created by{' '}
{currentProject.creator?.displayName ||
currentProject.creator?.email}{' '}
{formatDistanceToNowStrict(
parseISO(currentProject.createdAt),
)}{' '}
ago
</Text>
)}
<div className="mt-1 inline-grid grid-flow-col items-center justify-start gap-2 md:mt-0">
<Chip
size="small"
label={isPro ? 'Pro' : 'Starter'}
color={isPro ? 'primary' : 'default'}
/>
{!isPro && (
<Button
variant="borderless"
className="mr-2"
onClick={() => {
openAlertDialog({
title: 'Upgrade your plan.',
payload: <ChangePlanModal />,
props: {
PaperProps: { className: 'p-0 max-w-xl w-full' },
hidePrimaryAction: true,
hideSecondaryAction: true,
hideTitle: true,
},
});
}}
>
Upgrade
</Button>
)}
</div>
</div>
{currentProject.creator && (
<Text
color="secondary"
variant="subtitle2"
className="hidden md:block"
>
Created by{' '}
{currentProject.creator?.displayName ||
currentProject.creator?.email}{' '}
{formatDistanceToNowStrict(parseISO(currentProject.createdAt))}{' '}
ago
</Text>
)}
</div>
</div>
</div>
<Link
href={`/${currentWorkspace.slug}/${currentApplication.slug}/settings/general`}
href={`/${currentWorkspace.slug}/${currentProject.slug}/settings/general`}
passHref
>
<Button

View File

@@ -0,0 +1,201 @@
import Form from '@/components/common/Form';
import SettingsContainer from '@/components/settings/SettingsContainer';
import BaseProviderSettings from '@/components/settings/signInMethods/BaseProviderSettings';
import { useUI } from '@/context/UIContext';
import {
GetSignInMethodsDocument,
useGetSignInMethodsQuery,
useUpdateConfigMutation,
} from '@/generated/graphql';
import { useCurrentWorkspaceAndApplication } from '@/hooks/useCurrentWorkspaceAndApplication';
import ActivityIndicator from '@/ui/v2/ActivityIndicator';
import IconButton from '@/ui/v2/IconButton';
import CopyIcon from '@/ui/v2/icons/CopyIcon';
import Input from '@/ui/v2/Input';
import InputAdornment from '@/ui/v2/InputAdornment';
import generateAppServiceUrl from '@/utils/common/generateAppServiceUrl';
import { copy } from '@/utils/copy';
import getServerError from '@/utils/settings/getServerError';
import { getToastStyleProps } from '@/utils/settings/settingsConstants';
import { yupResolver } from '@hookform/resolvers/yup';
import { FormProvider, useForm } from 'react-hook-form';
import { toast } from 'react-hot-toast';
import { twMerge } from 'tailwind-merge';
import * as Yup from 'yup';
const validationSchema = Yup.object({
clientId: Yup.string()
.label('Client ID')
.when('enabled', {
is: true,
then: (schema) => schema.required(),
}),
clientSecret: Yup.string()
.label('Client Secret')
.when('enabled', {
is: true,
then: (schema) => schema.required(),
}),
tenant: Yup.string()
.label('Tenant')
.when('enabled', {
is: true,
then: (schema) => schema.required(),
}),
enabled: Yup.boolean(),
});
export type AzureADProviderFormValues = Yup.InferType<typeof validationSchema>;
export default function AzureADProviderSettings() {
const { maintenanceActive } = useUI();
const { currentApplication } = useCurrentWorkspaceAndApplication();
const [updateConfig] = useUpdateConfigMutation({
refetchQueries: [GetSignInMethodsDocument],
});
const { data, loading, error } = useGetSignInMethodsQuery({
variables: { appId: currentApplication?.id },
fetchPolicy: 'cache-only',
});
const { clientId, clientSecret, tenant, enabled } =
data?.config?.auth?.method?.oauth?.azuread || {};
const form = useForm<AzureADProviderFormValues>({
reValidateMode: 'onSubmit',
defaultValues: {
clientId: clientId || '',
clientSecret: clientSecret || '',
tenant: tenant || '',
enabled: enabled || false,
},
resolver: yupResolver(validationSchema),
});
if (loading) {
return (
<ActivityIndicator
delay={1000}
label="Loading settings for Azure AD..."
className="justify-center"
/>
);
}
if (error) {
throw error;
}
const { register, formState, watch } = form;
const authEnabled = watch('enabled');
const handleProviderUpdate = async (values: AzureADProviderFormValues) => {
const updateConfigPromise = updateConfig({
variables: {
appId: currentApplication.id,
config: {
auth: {
method: {
oauth: {
azuread: values,
},
},
},
},
},
});
try {
await toast.promise(
updateConfigPromise,
{
loading: `Azure AD settings are being updated...`,
success: `Azure AD settings have been updated successfully.`,
error: getServerError(
`An error occurred while trying to update the project's Azure AD settings.`,
),
},
getToastStyleProps(),
);
form.reset(values);
} catch {
// Note: The toast will handle the error.
}
};
return (
<FormProvider {...form}>
<Form onSubmit={handleProviderUpdate}>
<SettingsContainer
title="Azure AD"
description="Allow users to sign in with Azure AD."
slotProps={{
submitButton: {
disabled: !formState.isDirty || maintenanceActive,
loading: formState.isSubmitting,
},
}}
icon="/assets/brands/azuread.svg"
switchId="enabled"
showSwitch
className={twMerge(
'grid grid-flow-row grid-cols-2 gap-y-4 gap-x-3 px-4 py-2',
!authEnabled && 'hidden',
)}
>
<BaseProviderSettings providerName="azuread" />
<Input
{...register('tenant')}
name="tenant"
id="tenant"
label="Tenant ID"
placeholder="Tenant ID"
className="col-span-2"
fullWidth
hideEmptyHelperText
error={!!formState.errors?.tenant}
helperText={formState.errors?.tenant?.message}
/>
<Input
name="redirectUrl"
id="redirectUrl"
defaultValue={`${generateAppServiceUrl(
currentApplication.subdomain,
currentApplication.region.awsName,
'auth',
)}/signin/provider/azuread/callback`}
className="col-span-2"
fullWidth
hideEmptyHelperText
label="Redirect URL"
disabled
endAdornment={
<InputAdornment position="end" className="absolute right-2">
<IconButton
sx={{ minWidth: 0, padding: 0 }}
color="secondary"
variant="borderless"
onClick={(e) => {
e.stopPropagation();
copy(
`${generateAppServiceUrl(
currentApplication.subdomain,
currentApplication.region.awsName,
'auth',
)}/signin/provider/azuread/callback`,
'Redirect URL',
);
}}
>
<CopyIcon className="h-4 w-4" />
</IconButton>
</InputAdornment>
}
/>
</SettingsContainer>
</Form>
</FormProvider>
);
}

View File

@@ -0,0 +1 @@
export { default } from './AzureADProviderSettings';

View File

@@ -11,7 +11,7 @@ const StyledListItemText = styled(MaterialListItemText)(({ theme }) => ({
display: 'grid',
justifyContent: 'start',
gridAutoFlow: 'row',
gap: theme.spacing(0.5),
gap: theme.spacing(0.25),
fontSize: theme.typography.pxToRem(15),
[`&.${listItemTextClasses.root}`]: {
margin: 0,

View File

@@ -32,7 +32,6 @@ export const validationSchema = Yup.object({
.required('This field is required.'),
password: Yup.string()
.label('Users Password')
.min(8, 'Password must be at least 8 characters long.')
.required('This field is required.'),
});
@@ -99,7 +98,7 @@ export default function CreateUserForm({
}),
{
loading: 'Creating user...',
success: 'User created successfully.',
success: 'User has been created successfully.',
error: getServerError(
'An error occurred while trying to create the user.',
),

View File

@@ -19,6 +19,7 @@ import Option from '@/ui/v2/Option';
import Text from '@/ui/v2/Text';
import getReadableProviderName from '@/utils/common/getReadableProviderName';
import { copy } from '@/utils/copy';
import getServerError from '@/utils/settings/getServerError';
import getUserRoles from '@/utils/settings/getUserRoles';
import { getToastStyleProps } from '@/utils/settings/settingsConstants';
import {
@@ -168,11 +169,13 @@ export default function EditUserForm({
{
loading: shouldBan ? 'Banning user...' : 'Unbanning user...',
success: shouldBan
? 'User banned successfully'
: 'User unbanned successfully.',
error: shouldBan
? 'An error occurred while trying to ban the user.'
: 'An error occurred while trying to unban the user.',
? 'User has been banned successfully.'
: 'User has been unbanned successfully.',
error: getServerError(
shouldBan
? 'An error occurred while trying to ban the user.'
: 'An error occurred while trying to unban the user.',
),
},
getToastStyleProps(),
);
@@ -213,7 +216,7 @@ export default function EditUserForm({
Actions
</Button>
</Dropdown.Trigger>
<Dropdown.Content menu disablePortal className="h-full w-full">
<Dropdown.Content menu className="h-full w-full">
<Dropdown.Item
className="font-medium"
sx={{ color: 'error.main' }}
@@ -316,6 +319,7 @@ export default function EditUserForm({
id="emailVerified"
name="emailVerified"
label="Verified"
aria-label="Email Verified"
/>
)
}
@@ -354,6 +358,7 @@ export default function EditUserForm({
id="phoneNumberVerified"
name="phoneNumberVerified"
label="Verified"
aria-label="Phone Number Verified"
disabled={!form.watch('phoneNumber')}
/>
)

View File

@@ -151,7 +151,7 @@ export default function UsersBody({ users, onSubmit }: UsersBodyProps) {
updateUserMutationPromise,
{
loading: `Updating user's settings...`,
success: 'User settings updated successfully.',
success: 'User settings have been updated successfully.',
error: getServerError(
`An error occurred while trying to update this user's settings.`,
),

View File

@@ -1,21 +1,31 @@
import { useCurrentWorkspaceAndApplication } from '@/hooks/useCurrentWorkspaceAndApplication';
import Status, { StatusEnum } from '@/ui/Status';
import RetryableErrorBoundary from '@/components/common/RetryableErrorBoundary';
import { useCurrentWorkspaceAndProject } from '@/hooks/v2/useCurrentWorkspaceAndProject';
import ActivityIndicator from '@/ui/v2/ActivityIndicator';
import Box from '@/ui/v2/Box';
import Button from '@/ui/v2/Button';
import Chip from '@/ui/v2/Chip';
import Divider from '@/ui/v2/Divider';
import PlusCircleIcon from '@/ui/v2/icons/PlusCircleIcon';
import List from '@/ui/v2/List';
import { ListItem } from '@/ui/v2/ListItem';
import Text from '@/ui/v2/Text';
import { formatDistanceToNowStrict, parseISO } from 'date-fns';
import Image from 'next/image';
import NavLink from 'next/link';
import { Fragment } from 'react';
function AllWorkspaceApps() {
const { currentWorkspace } = useCurrentWorkspaceAndApplication();
const noApplications = currentWorkspace?.applications.length === 0;
const { currentWorkspace, loading, error } = useCurrentWorkspaceAndProject();
if (noApplications) {
if (loading) {
return <ActivityIndicator label="Loading projects..." delay={1000} />;
}
if (error) {
throw error;
}
if (currentWorkspace?.projects?.length === 0) {
return (
<Box className="flex flex-row border-y py-4">
<Text className="text-xs" color="secondary">
@@ -29,25 +39,50 @@ function AllWorkspaceApps() {
<List>
<Divider component="li" />
{currentWorkspace?.applications.map((app) => (
<Fragment key={app.id}>
{currentWorkspace?.projects.map((project) => (
<Fragment key={project.id}>
<ListItem.Root>
<NavLink href={`${currentWorkspace?.slug}/${app.slug}`} passHref>
<ListItem.Button className="grid grid-flow-col justify-between gap-2">
<div className="grid grid-flow-col items-center gap-2">
<div className="h-8 w-8 overflow-hidden rounded-lg">
<Image
src="/logos/new.svg"
alt="Nhost Logo"
width={32}
height={32}
/>
</div>
<NavLink
href={`${currentWorkspace?.slug}/${project.slug}`}
passHref
>
<ListItem.Button className="grid grid-flow-col items-center justify-between gap-2">
<div className="grid grid-flow-col items-center justify-start gap-2">
<ListItem.Avatar>
<div className="h-8 w-8 overflow-hidden rounded-lg">
<Image
src="/logos/new.svg"
alt="Nhost Logo"
width={32}
height={32}
/>
</div>
</ListItem.Avatar>
<Text className="font-medium">{app.name}</Text>
<ListItem.Text
primary={project.name}
secondary={
project.creator ? (
<span>
{`Created by ${
project.creator.displayName || project.creator.email
} ${formatDistanceToNowStrict(
parseISO(project.createdAt),
)} ago`}
</span>
) : undefined
}
secondaryTypographyProps={{
className: 'text-xs',
}}
/>
</div>
<Status status={StatusEnum.Plan}>{app.plan.name}</Status>
<Chip
size="small"
label={project.plan.isFree ? 'Starter' : 'Pro'}
color={project.plan.isFree ? 'default' : 'primary'}
/>
</ListItem.Button>
</NavLink>
</ListItem.Root>
@@ -59,30 +94,35 @@ function AllWorkspaceApps() {
);
}
export default function WorkspaceApps() {
const { currentWorkspace } = useCurrentWorkspaceAndApplication();
const { currentWorkspace, loading } = useCurrentWorkspaceAndProject();
return (
<div className="mt-9">
<div className="mx-auto max-w-3xl font-display">
<div className="mb-4 flex flex-row place-content-between">
<div className="mb-4 grid grid-flow-col items-center justify-between gap-2">
<Text className="text-lg font-medium">Projects</Text>
<NavLink
href={{
pathname: '/new',
query: { workspace: currentWorkspace.slug },
}}
>
<Button
variant="outlined"
color="secondary"
startIcon={<PlusCircleIcon />}
{!loading && (
<NavLink
href={{
pathname: '/new',
query: { workspace: currentWorkspace?.slug },
}}
>
New Project
</Button>
</NavLink>
<Button
variant="outlined"
color="secondary"
startIcon={<PlusCircleIcon />}
>
New Project
</Button>
</NavLink>
)}
</div>
<AllWorkspaceApps />
<RetryableErrorBoundary errorMessageProps={{ className: 'px-0' }}>
<AllWorkspaceApps />
</RetryableErrorBoundary>
</div>
</div>
);

View File

@@ -0,0 +1,26 @@
fragment Workspace on workspaces {
id
name
slug
workspaceMembers {
id
user {
id
email
displayName
}
type
}
projects: apps {
...Project
}
}
query GetWorkspaceAndProject($workspaceSlug: String!, $projectSlug: String) {
workspaces(where: { slug: { _eq: $workspaceSlug } }) {
...Workspace
}
projects: apps(where: { slug: { _eq: $projectSlug } }) {
...Project
}
}

View File

@@ -100,6 +100,12 @@ query GetSignInMethods($appId: uuid!) {
connection
organization
}
azuread {
enabled
clientId
clientSecret
tenant
}
}
}
}

View File

@@ -0,0 +1,59 @@
fragment Project on apps {
id
slug
name
repositoryProductionBranch
subdomain
isProvisioned
createdAt
desiredState
nhostBaseFolder
providersUpdated
config(resolve: true) {
hasura {
adminSecret
}
}
featureFlags {
description
id
name
value
}
appStates(order_by: { createdAt: desc }, limit: 1) {
id
appId
message
stateId
createdAt
}
region {
id
countryCode
awsName
city
}
plan {
id
name
isFree
}
githubRepository {
fullName
}
deployments(limit: 4, order_by: { deploymentEndedAt: desc }) {
id
commitSHA
commitMessage
commitUserName
deploymentStartedAt
deploymentEndedAt
commitUserAvatarUrl
deploymentStatus
}
creator {
id
email
displayName
}
}

View File

@@ -1,58 +1,3 @@
fragment Project on apps {
id
slug
name
repositoryProductionBranch
subdomain
isProvisioned
createdAt
desiredState
nhostBaseFolder
providersUpdated
config(resolve: true) {
hasura {
adminSecret
}
}
featureFlags {
description
id
name
value
}
appStates(order_by: { createdAt: desc }, limit: 1) {
id
appId
message
stateId
createdAt
}
region {
id
countryCode
awsName
city
}
plan {
id
name
isFree
}
githubRepository {
fullName
}
deployments(limit: 4, order_by: { deploymentEndedAt: desc }) {
id
commitSHA
commitMessage
commitUserName
deploymentStartedAt
deploymentEndedAt
commitUserAvatarUrl
deploymentStatus
}
}
query getOneUser($userId: uuid!) {
user(id: $userId) {
id

View File

@@ -39,6 +39,8 @@ export function useCheckProvisioning() {
const memoizedUpdateCache = useCallback(updateOwnCache, [client]);
const currentApplicationId = currentApplication?.id;
useEffect(() => {
startPolling(2000);
}, [startPolling]);
@@ -84,7 +86,7 @@ export function useCheckProvisioning() {
createdAt: data.app.appStates[0].createdAt,
});
discordAnnounce(
`Application ${currentApplication.id} errored after provisioning: ${data.app.appStates[0].message}`,
`Application ${currentApplicationId} errored after provisioning: ${data.app.appStates[0].message}`,
);
stopPolling();
memoizedUpdateCache();
@@ -93,7 +95,7 @@ export function useCheckProvisioning() {
data,
stopPolling,
memoizedUpdateCache,
currentApplication.id,
currentApplicationId,
currentApplicationState.state,
]);

View File

@@ -0,0 +1 @@
export { default as useCurrentWorkspaceAndProject } from './useCurrentWorkspaceAndProject';

View File

@@ -0,0 +1,115 @@
import useIsPlatform from '@/hooks/common/useIsPlatform';
import type { Project, Workspace } from '@/types/application';
import { ApplicationStatus } from '@/types/application';
import { getHasuraAdminSecret } from '@/utils/env';
import type { GetWorkspaceAndProjectQueryHookResult } from '@/utils/__generated__/graphql';
import { useGetWorkspaceAndProjectQuery } from '@/utils/__generated__/graphql';
import type { QueryHookOptions } from '@apollo/client';
import { useRouter } from 'next/router';
export interface UseCurrentWorkspaceAndProjectOptions {
/**
* The fetch policy to use.
*
* @default 'cache-first'
*/
fetchPolicy?: QueryHookOptions['fetchPolicy'];
}
export interface UseCurrentWorkspaceAndProjectReturnType {
/**
* The current workspace.
*/
currentWorkspace: Workspace;
/**
* The current project.
*/
currentProject: Project;
/**
* Whether the query is loading.
*/
loading?: GetWorkspaceAndProjectQueryHookResult['loading'];
/**
* The error if any.
*/
error?: GetWorkspaceAndProjectQueryHookResult['error'];
}
export default function useCurrentWorkspaceAndProject(
options?: UseCurrentWorkspaceAndProjectOptions,
): UseCurrentWorkspaceAndProjectReturnType {
const isPlatform = useIsPlatform();
const {
query: { workspaceSlug, appSlug },
isReady,
} = useRouter();
const { data, loading, error } = useGetWorkspaceAndProjectQuery({
variables: {
workspaceSlug: (workspaceSlug as string) || '',
projectSlug: (appSlug as string) || '',
},
fetchPolicy: options?.fetchPolicy || 'cache-first',
skip: !isPlatform || !isReady || !workspaceSlug,
});
if (!isPlatform) {
const localProject: Project = {
id: 'local',
slug: 'local',
name: 'local',
appStates: [
{
id: 'local',
appId: 'local',
stateId: ApplicationStatus.Live,
createdAt: new Date().toISOString(),
},
],
deployments: [],
subdomain: 'local',
region: {
id: null,
countryCode: null,
city: null,
awsName: null,
},
isProvisioned: true,
createdAt: new Date().toISOString(),
desiredState: ApplicationStatus.Live,
featureFlags: [],
providersUpdated: true,
repositoryProductionBranch: null,
nhostBaseFolder: null,
plan: null,
config: {
hasura: {
adminSecret: getHasuraAdminSecret(),
},
},
};
return {
currentWorkspace: {
id: 'local',
slug: 'local',
name: 'local',
projects: [localProject],
workspaceMembers: [],
},
currentProject: localProject,
loading: false,
};
}
const [currentWorkspace] = data?.workspaces || [];
const [currentProject] = data?.projects || [];
return {
currentWorkspace,
currentProject,
loading: data ? false : loading,
error,
};
}

View File

@@ -16,9 +16,7 @@ export default function DataBrowserDatabaseDetailsPage() {
description={
<span>
Database{' '}
<InlineCode className="bg-gray-200 bg-opacity-80 px-1.5 text-sm">
{dataSourceSlug}
</InlineCode>{' '}
<InlineCode className="px-1.5 text-sm">{dataSourceSlug}</InlineCode>{' '}
does not exist.
</span>
}

View File

@@ -134,9 +134,5 @@ export default function LogsPage() {
}
LogsPage.getLayout = function getLayout(page: ReactElement) {
return (
<ProjectLayout mainContainerProps={{ className: 'bg-gray-50' }}>
{page}
</ProjectLayout>
);
return <ProjectLayout>{page}</ProjectLayout>;
};

View File

@@ -2,6 +2,7 @@ import Container from '@/components/layout/Container';
import SettingsLayout from '@/components/settings/SettingsLayout';
import AnonymousSignInSettings from '@/components/settings/signInMethods/AnonymousSignInSettings';
import AppleProviderSettings from '@/components/settings/signInMethods/AppleProviderSettings';
import AzureADProviderSettings from '@/components/settings/signInMethods/AzureADProviderSettings';
import DiscordProviderSettings from '@/components/settings/signInMethods/DiscordProviderSettings';
import EmailAndPasswordSettings from '@/components/settings/signInMethods/EmailAndPasswordSettings';
import FacebookProviderSettings from '@/components/settings/signInMethods/FacebookProviderSettings';
@@ -56,6 +57,7 @@ export default function SettingsSignInMethodsPage() {
<SMSSettings />
{!currentApplication.providersUpdated && <ProvidersUpdatedAlert />}
<AppleProviderSettings />
<AzureADProviderSettings />
<DiscordProviderSettings />
<FacebookProviderSettings />
<GitHubProviderSettings />

View File

@@ -6,6 +6,7 @@ import type {
PermissionVariableFragment,
ProjectFragment,
SecretFragment,
WorkspaceFragment,
} from '@/utils/__generated__/graphql';
/**
@@ -62,6 +63,7 @@ export type FeatureFlag = {
value: string;
};
export type Workspace = WorkspaceFragment;
export type Project = ProjectFragment;
export interface PermissionVariable extends PermissionVariableFragment {

View File

@@ -25,7 +25,9 @@ export type Scalars = {
bpchar: any;
bytea: any;
citext: any;
float64: any;
jsonb: any;
labels: any;
smallint: any;
timestamp: any;
timestamptz: any;
@@ -1035,7 +1037,9 @@ export type ConfigGlobalUpdateInput = {
export type ConfigHasura = {
__typename?: 'ConfigHasura';
adminSecret: Scalars['String'];
events?: Maybe<ConfigHasuraEvents>;
jwtSecrets?: Maybe<Array<ConfigJwtSecret>>;
logs?: Maybe<ConfigHasuraLogs>;
resources?: Maybe<ConfigResources>;
settings?: Maybe<ConfigHasuraSettings>;
version?: Maybe<Scalars['String']>;
@@ -1047,22 +1051,66 @@ export type ConfigHasuraComparisonExp = {
_not?: InputMaybe<ConfigHasuraComparisonExp>;
_or?: InputMaybe<Array<ConfigHasuraComparisonExp>>;
adminSecret?: InputMaybe<ConfigStringComparisonExp>;
events?: InputMaybe<ConfigHasuraEventsComparisonExp>;
jwtSecrets?: InputMaybe<ConfigJwtSecretComparisonExp>;
logs?: InputMaybe<ConfigHasuraLogsComparisonExp>;
resources?: InputMaybe<ConfigResourcesComparisonExp>;
settings?: InputMaybe<ConfigHasuraSettingsComparisonExp>;
version?: InputMaybe<ConfigStringComparisonExp>;
webhookSecret?: InputMaybe<ConfigStringComparisonExp>;
};
export type ConfigHasuraEvents = {
__typename?: 'ConfigHasuraEvents';
httpPoolSize?: Maybe<Scalars['ConfigUint32']>;
};
export type ConfigHasuraEventsComparisonExp = {
_and?: InputMaybe<Array<ConfigHasuraEventsComparisonExp>>;
_not?: InputMaybe<ConfigHasuraEventsComparisonExp>;
_or?: InputMaybe<Array<ConfigHasuraEventsComparisonExp>>;
httpPoolSize?: InputMaybe<ConfigUint32ComparisonExp>;
};
export type ConfigHasuraEventsInsertInput = {
httpPoolSize?: InputMaybe<Scalars['ConfigUint32']>;
};
export type ConfigHasuraEventsUpdateInput = {
httpPoolSize?: InputMaybe<Scalars['ConfigUint32']>;
};
export type ConfigHasuraInsertInput = {
adminSecret: Scalars['String'];
events?: InputMaybe<ConfigHasuraEventsInsertInput>;
jwtSecrets?: InputMaybe<Array<ConfigJwtSecretInsertInput>>;
logs?: InputMaybe<ConfigHasuraLogsInsertInput>;
resources?: InputMaybe<ConfigResourcesInsertInput>;
settings?: InputMaybe<ConfigHasuraSettingsInsertInput>;
version?: InputMaybe<Scalars['String']>;
webhookSecret: Scalars['String'];
};
export type ConfigHasuraLogs = {
__typename?: 'ConfigHasuraLogs';
level?: Maybe<Scalars['String']>;
};
export type ConfigHasuraLogsComparisonExp = {
_and?: InputMaybe<Array<ConfigHasuraLogsComparisonExp>>;
_not?: InputMaybe<ConfigHasuraLogsComparisonExp>;
_or?: InputMaybe<Array<ConfigHasuraLogsComparisonExp>>;
level?: InputMaybe<ConfigStringComparisonExp>;
};
export type ConfigHasuraLogsInsertInput = {
level?: InputMaybe<Scalars['String']>;
};
export type ConfigHasuraLogsUpdateInput = {
level?: InputMaybe<Scalars['String']>;
};
export type ConfigHasuraSettings = {
__typename?: 'ConfigHasuraSettings';
enableRemoteSchemaPermissions?: Maybe<Scalars['Boolean']>;
@@ -1085,7 +1133,9 @@ export type ConfigHasuraSettingsUpdateInput = {
export type ConfigHasuraUpdateInput = {
adminSecret?: InputMaybe<Scalars['String']>;
events?: InputMaybe<ConfigHasuraEventsUpdateInput>;
jwtSecrets?: InputMaybe<Array<ConfigJwtSecretUpdateInput>>;
logs?: InputMaybe<ConfigHasuraLogsUpdateInput>;
resources?: InputMaybe<ConfigResourcesUpdateInput>;
settings?: InputMaybe<ConfigHasuraSettingsUpdateInput>;
version?: InputMaybe<Scalars['String']>;
@@ -1639,6 +1689,24 @@ export type Log = {
timestamp: Scalars['Timestamp'];
};
export type Metrics = {
__typename?: 'Metrics';
rows: Array<RowMetric>;
};
export type RowMetric = {
__typename?: 'RowMetric';
labels?: Maybe<Scalars['labels']>;
time: Scalars['Timestamp'];
value?: Maybe<Scalars['float64']>;
};
export type StatsLiveApps = {
__typename?: 'StatsLiveApps';
appID: Array<Scalars['uuid']>;
count: Scalars['Int'];
};
/** Boolean expression to compare columns of type "String". All fields are combined with logical 'AND'. */
export type String_Comparison_Exp = {
_eq?: InputMaybe<Scalars['String']>;
@@ -4878,6 +4946,7 @@ export type Backups = {
appId: Scalars['uuid'];
completedAt?: Maybe<Scalars['timestamptz']>;
createdAt: Scalars['timestamptz'];
expiresAt?: Maybe<Scalars['timestamptz']>;
id: Scalars['uuid'];
size: Scalars['bigint'];
};
@@ -4965,6 +5034,7 @@ export type Backups_Bool_Exp = {
appId?: InputMaybe<Uuid_Comparison_Exp>;
completedAt?: InputMaybe<Timestamptz_Comparison_Exp>;
createdAt?: InputMaybe<Timestamptz_Comparison_Exp>;
expiresAt?: InputMaybe<Timestamptz_Comparison_Exp>;
id?: InputMaybe<Uuid_Comparison_Exp>;
size?: InputMaybe<Bigint_Comparison_Exp>;
};
@@ -4986,6 +5056,7 @@ export type Backups_Insert_Input = {
appId?: InputMaybe<Scalars['uuid']>;
completedAt?: InputMaybe<Scalars['timestamptz']>;
createdAt?: InputMaybe<Scalars['timestamptz']>;
expiresAt?: InputMaybe<Scalars['timestamptz']>;
id?: InputMaybe<Scalars['uuid']>;
size?: InputMaybe<Scalars['bigint']>;
};
@@ -4996,6 +5067,7 @@ export type Backups_Max_Fields = {
appId?: Maybe<Scalars['uuid']>;
completedAt?: Maybe<Scalars['timestamptz']>;
createdAt?: Maybe<Scalars['timestamptz']>;
expiresAt?: Maybe<Scalars['timestamptz']>;
id?: Maybe<Scalars['uuid']>;
size?: Maybe<Scalars['bigint']>;
};
@@ -5005,6 +5077,7 @@ export type Backups_Max_Order_By = {
appId?: InputMaybe<Order_By>;
completedAt?: InputMaybe<Order_By>;
createdAt?: InputMaybe<Order_By>;
expiresAt?: InputMaybe<Order_By>;
id?: InputMaybe<Order_By>;
size?: InputMaybe<Order_By>;
};
@@ -5015,6 +5088,7 @@ export type Backups_Min_Fields = {
appId?: Maybe<Scalars['uuid']>;
completedAt?: Maybe<Scalars['timestamptz']>;
createdAt?: Maybe<Scalars['timestamptz']>;
expiresAt?: Maybe<Scalars['timestamptz']>;
id?: Maybe<Scalars['uuid']>;
size?: Maybe<Scalars['bigint']>;
};
@@ -5024,6 +5098,7 @@ export type Backups_Min_Order_By = {
appId?: InputMaybe<Order_By>;
completedAt?: InputMaybe<Order_By>;
createdAt?: InputMaybe<Order_By>;
expiresAt?: InputMaybe<Order_By>;
id?: InputMaybe<Order_By>;
size?: InputMaybe<Order_By>;
};
@@ -5050,6 +5125,7 @@ export type Backups_Order_By = {
appId?: InputMaybe<Order_By>;
completedAt?: InputMaybe<Order_By>;
createdAt?: InputMaybe<Order_By>;
expiresAt?: InputMaybe<Order_By>;
id?: InputMaybe<Order_By>;
size?: InputMaybe<Order_By>;
};
@@ -5068,6 +5144,8 @@ export enum Backups_Select_Column {
/** column name */
CreatedAt = 'createdAt',
/** column name */
ExpiresAt = 'expiresAt',
/** column name */
Id = 'id',
/** column name */
Size = 'size'
@@ -5078,6 +5156,7 @@ export type Backups_Set_Input = {
appId?: InputMaybe<Scalars['uuid']>;
completedAt?: InputMaybe<Scalars['timestamptz']>;
createdAt?: InputMaybe<Scalars['timestamptz']>;
expiresAt?: InputMaybe<Scalars['timestamptz']>;
id?: InputMaybe<Scalars['uuid']>;
size?: InputMaybe<Scalars['bigint']>;
};
@@ -5128,6 +5207,7 @@ export type Backups_Stream_Cursor_Value_Input = {
appId?: InputMaybe<Scalars['uuid']>;
completedAt?: InputMaybe<Scalars['timestamptz']>;
createdAt?: InputMaybe<Scalars['timestamptz']>;
expiresAt?: InputMaybe<Scalars['timestamptz']>;
id?: InputMaybe<Scalars['uuid']>;
size?: InputMaybe<Scalars['bigint']>;
};
@@ -5152,6 +5232,8 @@ export enum Backups_Update_Column {
/** column name */
CreatedAt = 'createdAt',
/** column name */
ExpiresAt = 'expiresAt',
/** column name */
Id = 'id',
/** column name */
Size = 'size'
@@ -9150,6 +9232,7 @@ export type Mutation_Root = {
/** insert a single row into the table: "regions" */
insert_regions_one?: Maybe<Regions>;
migrateRDSToPostgres: Scalars['Boolean'];
pauseInactiveApps: Array<Scalars['String']>;
resetPostgresPassword: Scalars['Boolean'];
restoreApplicationDatabase: Scalars['Boolean'];
/** update single row of the table: "apps" */
@@ -9338,9 +9421,16 @@ export type Mutation_Root = {
};
/** mutation root */
export type Mutation_RootBackupAllApplicationsDatabaseArgs = {
expireInDays?: InputMaybe<Scalars['Int']>;
};
/** mutation root */
export type Mutation_RootBackupApplicationDatabaseArgs = {
appID: Scalars['String'];
expireInDays?: InputMaybe<Scalars['Int']>;
};
@@ -11947,6 +12037,41 @@ export type Query_Root = {
files: Array<Files>;
/** fetch aggregated fields from the table: "storage.files" */
filesAggregate: Files_Aggregate;
/**
* Returns CPU metrics for a given application.
* If `from` and `to` are not provided, they default to an hour ago and now, respectively.
*
* CPU usage is calculated as the average CPU usage over the period of 1m.
*
* Unit returned is millicores.
*/
getCPUMetrics: Metrics;
/**
* Returns memory metrics for a given application.
* If `from` and `to` are not provided, they default to an hour ago and now, respectively.
*
* Memory usage is returned in MiB.
*/
getMemoryMetrics: Metrics;
/**
* Returns disk capacity for the volume used by postgres to store the database.
* If `from` and `to` are not provided, they default to an hour ago and now, respectively.
*
* Disk usage is returned in MiB.
*/
getPostgresVolumeCapacity: Metrics;
/**
* Returns disk usage for the volume used by postgres to store the database.
* If `from` and `to` are not provided, they default to an hour ago and now, respectively.
*
* Disk usage is returned in MiB.
*/
getPostgresVolumeUsage: Metrics;
/**
* Return requests per second for a given application by service.
* If `from` and `to` are not provided, they default to an hour ago and now, respectively.
*/
getRequestsPerSecond: Metrics;
/** fetch data from the table: "github_app_installations" using primary key columns */
githubAppInstallation?: Maybe<GithubAppInstallations>;
/** fetch data from the table: "github_app_installations" */
@@ -11982,6 +12107,13 @@ export type Query_Root = {
regions_aggregate: Regions_Aggregate;
/** fetch data from the table: "regions" using primary key columns */
regions_by_pk?: Maybe<Regions>;
/**
* Returns lists of apps that have some live traffic in the give time range.
* From defaults to 24 hours ago and to defaults to now.
*
* Requests that returned a 4xx or 5xx status code are not counted as live traffic.
*/
statsLiveApps: StatsLiveApps;
systemConfig?: Maybe<ConfigSystemConfig>;
systemConfigs: Array<ConfigAppSystemConfig>;
/** fetch data from the table: "auth.users" using primary key columns */
@@ -12511,6 +12643,41 @@ export type Query_RootFilesAggregateArgs = {
};
export type Query_RootGetCpuMetricsArgs = {
appID: Scalars['String'];
from?: InputMaybe<Scalars['Timestamp']>;
to?: InputMaybe<Scalars['Timestamp']>;
};
export type Query_RootGetMemoryMetricsArgs = {
appID: Scalars['String'];
from?: InputMaybe<Scalars['Timestamp']>;
to?: InputMaybe<Scalars['Timestamp']>;
};
export type Query_RootGetPostgresVolumeCapacityArgs = {
appID: Scalars['String'];
from?: InputMaybe<Scalars['Timestamp']>;
to?: InputMaybe<Scalars['Timestamp']>;
};
export type Query_RootGetPostgresVolumeUsageArgs = {
appID: Scalars['String'];
from?: InputMaybe<Scalars['Timestamp']>;
to?: InputMaybe<Scalars['Timestamp']>;
};
export type Query_RootGetRequestsPerSecondArgs = {
appID: Scalars['String'];
from?: InputMaybe<Scalars['Timestamp']>;
to?: InputMaybe<Scalars['Timestamp']>;
};
export type Query_RootGithubAppInstallationArgs = {
id: Scalars['uuid'];
};
@@ -12634,6 +12801,12 @@ export type Query_RootRegions_By_PkArgs = {
};
export type Query_RootStatsLiveAppsArgs = {
from?: InputMaybe<Scalars['Timestamp']>;
to?: InputMaybe<Scalars['Timestamp']>;
};
export type Query_RootSystemConfigArgs = {
appID: Scalars['uuid'];
};
@@ -16390,6 +16563,16 @@ export type GetRemoteAppRolesQueryVariables = Exact<{ [key: string]: never; }>;
export type GetRemoteAppRolesQuery = { __typename?: 'query_root', authRoles: Array<{ __typename?: 'authRoles', role: string }> };
export type WorkspaceFragment = { __typename?: 'workspaces', id: any, name: string, slug: string, workspaceMembers: Array<{ __typename?: 'workspaceMembers', id: any, type: string, user: { __typename?: 'users', id: any, email?: any | null, displayName: string } }>, projects: Array<{ __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }>, creator?: { __typename?: 'users', id: any, email?: any | null, displayName: string } | null }> };
export type GetWorkspaceAndProjectQueryVariables = Exact<{
workspaceSlug: Scalars['String'];
projectSlug?: InputMaybe<Scalars['String']>;
}>;
export type GetWorkspaceAndProjectQuery = { __typename?: 'query_root', workspaces: Array<{ __typename?: 'workspaces', id: any, name: string, slug: string, workspaceMembers: Array<{ __typename?: 'workspaceMembers', id: any, type: string, user: { __typename?: 'users', id: any, email?: any | null, displayName: string } }>, projects: Array<{ __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }>, creator?: { __typename?: 'users', id: any, email?: any | null, displayName: string } | null }> }>, projects: Array<{ __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }>, creator?: { __typename?: 'users', id: any, email?: any | null, displayName: string } | null }> };
export type InsertApplicationMutationVariables = Exact<{
app: Apps_Insert_Input;
}>;
@@ -16480,7 +16663,7 @@ export type GetSignInMethodsQueryVariables = Exact<{
}>;
export type GetSignInMethodsQuery = { __typename?: 'query_root', config?: { __typename: 'ConfigConfig', id: 'ConfigConfig', provider?: { __typename: 'ConfigProvider', id: 'ConfigProvider', sms?: { __typename?: 'ConfigSms', accountSid: string, authToken: string, messagingServiceId: string, provider?: string | null } | null } | null, auth?: { __typename: 'ConfigAuth', id: 'ConfigAuth', method?: { __typename?: 'ConfigAuthMethod', emailPassword?: { __typename?: 'ConfigAuthMethodEmailPassword', emailVerificationRequired?: boolean | null, hibpEnabled?: boolean | null } | null, emailPasswordless?: { __typename?: 'ConfigAuthMethodEmailPasswordless', enabled?: boolean | null } | null, smsPasswordless?: { __typename?: 'ConfigAuthMethodSmsPasswordless', enabled?: boolean | null } | null, anonymous?: { __typename?: 'ConfigAuthMethodAnonymous', enabled?: boolean | null } | null, webauthn?: { __typename?: 'ConfigAuthMethodWebauthn', enabled?: boolean | null } | null, oauth?: { __typename?: 'ConfigAuthMethodOauth', apple?: { __typename?: 'ConfigAuthMethodOauthApple', enabled?: boolean | null, clientId?: string | null, keyId?: string | null, teamId?: string | null, privateKey?: string | null } | null, discord?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, facebook?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, github?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, google?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, linkedin?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, spotify?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, twitch?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, twitter?: { __typename?: 'ConfigAuthMethodOauthTwitter', enabled?: boolean | null, consumerKey?: string | null, consumerSecret?: string | null } | null, windowslive?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, workos?: { __typename?: 'ConfigAuthMethodOauthWorkos', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, connection?: string | null, organization?: string | null } | null } | null } | null } | null } | null };
export type GetSignInMethodsQuery = { __typename?: 'query_root', config?: { __typename: 'ConfigConfig', id: 'ConfigConfig', provider?: { __typename: 'ConfigProvider', id: 'ConfigProvider', sms?: { __typename?: 'ConfigSms', accountSid: string, authToken: string, messagingServiceId: string, provider?: string | null } | null } | null, auth?: { __typename: 'ConfigAuth', id: 'ConfigAuth', method?: { __typename?: 'ConfigAuthMethod', emailPassword?: { __typename?: 'ConfigAuthMethodEmailPassword', emailVerificationRequired?: boolean | null, hibpEnabled?: boolean | null } | null, emailPasswordless?: { __typename?: 'ConfigAuthMethodEmailPasswordless', enabled?: boolean | null } | null, smsPasswordless?: { __typename?: 'ConfigAuthMethodSmsPasswordless', enabled?: boolean | null } | null, anonymous?: { __typename?: 'ConfigAuthMethodAnonymous', enabled?: boolean | null } | null, webauthn?: { __typename?: 'ConfigAuthMethodWebauthn', enabled?: boolean | null } | null, oauth?: { __typename?: 'ConfigAuthMethodOauth', apple?: { __typename?: 'ConfigAuthMethodOauthApple', enabled?: boolean | null, clientId?: string | null, keyId?: string | null, teamId?: string | null, privateKey?: string | null } | null, discord?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, facebook?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, github?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, google?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, linkedin?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, spotify?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, twitch?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, twitter?: { __typename?: 'ConfigAuthMethodOauthTwitter', enabled?: boolean | null, consumerKey?: string | null, consumerSecret?: string | null } | null, windowslive?: { __typename?: 'ConfigStandardOauthProviderWithScope', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, scope?: Array<string> | null } | null, workos?: { __typename?: 'ConfigAuthMethodOauthWorkos', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, connection?: string | null, organization?: string | null } | null, azuread?: { __typename?: 'ConfigAuthMethodOauthAzuread', enabled?: boolean | null, clientId?: string | null, clientSecret?: string | null, tenant?: string | null } | null } | null } | null } | null } | null };
export type GetSmtpSettingsQueryVariables = Exact<{
appId: Scalars['uuid'];
@@ -16617,6 +16800,8 @@ export type GetFilesAggregateQueryVariables = Exact<{
export type GetFilesAggregateQuery = { __typename?: 'query_root', filesAggregate: { __typename?: 'files_aggregate', aggregate?: { __typename?: 'files_aggregate_fields', count: number } | null } };
export type ProjectFragment = { __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }>, creator?: { __typename?: 'users', id: any, email?: any | null, displayName: string } | null };
export type GithubRepositoryFragment = { __typename?: 'githubRepositories', id: any, name: string, fullName: string, private: boolean, githubAppInstallation: { __typename?: 'githubAppInstallations', id: any, accountLogin?: string | null, accountType?: string | null, accountAvatarUrl?: string | null } };
export type GetGithubRepositoriesQueryVariables = Exact<{ [key: string]: never; }>;
@@ -16841,14 +17026,12 @@ export type GetFreeAndActiveProjectsQueryVariables = Exact<{
export type GetFreeAndActiveProjectsQuery = { __typename?: 'query_root', freeAndActiveProjects: Array<{ __typename?: 'apps', id: any }> };
export type ProjectFragment = { __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }> };
export type GetOneUserQueryVariables = Exact<{
userId: Scalars['uuid'];
}>;
export type GetOneUserQuery = { __typename?: 'query_root', user?: { __typename?: 'users', id: any, displayName: string, avatarUrl: string, workspaceMembers: Array<{ __typename?: 'workspaceMembers', id: any, userId: any, workspaceId: any, type: string, workspace: { __typename?: 'workspaces', creatorUserId?: any | null, id: any, slug: string, name: string, apps: Array<{ __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }> }> } }> } | null };
export type GetOneUserQuery = { __typename?: 'query_root', user?: { __typename?: 'users', id: any, displayName: string, avatarUrl: string, workspaceMembers: Array<{ __typename?: 'workspaceMembers', id: any, userId: any, workspaceId: any, type: string, workspace: { __typename?: 'workspaces', creatorUserId?: any | null, id: any, slug: string, name: string, apps: Array<{ __typename?: 'apps', id: any, slug: string, name: string, repositoryProductionBranch: string, subdomain: string, isProvisioned: boolean, createdAt: any, desiredState: number, nhostBaseFolder: string, providersUpdated?: boolean | null, config?: { __typename?: 'ConfigConfig', hasura: { __typename?: 'ConfigHasura', adminSecret: string } } | null, featureFlags: Array<{ __typename?: 'featureFlags', description: string, id: any, name: string, value: string }>, appStates: Array<{ __typename?: 'appStateHistory', id: any, appId: any, message?: string | null, stateId: number, createdAt: any }>, region: { __typename?: 'regions', id: any, countryCode: string, awsName: string, city: string }, plan: { __typename?: 'plans', id: any, name: string, isFree: boolean }, githubRepository?: { __typename?: 'githubRepositories', fullName: string } | null, deployments: Array<{ __typename?: 'deployments', id: any, commitSHA: string, commitMessage?: string | null, commitUserName?: string | null, deploymentStartedAt?: any | null, deploymentEndedAt?: any | null, commitUserAvatarUrl?: string | null, deploymentStatus?: string | null }>, creator?: { __typename?: 'users', id: any, email?: any | null, displayName: string } | null }> } }> } | null };
export type GetUserAllWorkspacesQueryVariables = Exact<{ [key: string]: never; }>;
@@ -17050,6 +17233,86 @@ export const GetAppByWorkspaceAndNameFragmentDoc = gql`
}
}
`;
export const ProjectFragmentDoc = gql`
fragment Project on apps {
id
slug
name
repositoryProductionBranch
subdomain
isProvisioned
createdAt
desiredState
nhostBaseFolder
providersUpdated
config(resolve: true) {
hasura {
adminSecret
}
}
featureFlags {
description
id
name
value
}
appStates(order_by: {createdAt: desc}, limit: 1) {
id
appId
message
stateId
createdAt
}
region {
id
countryCode
awsName
city
}
plan {
id
name
isFree
}
githubRepository {
fullName
}
deployments(limit: 4, order_by: {deploymentEndedAt: desc}) {
id
commitSHA
commitMessage
commitUserName
deploymentStartedAt
deploymentEndedAt
commitUserAvatarUrl
deploymentStatus
}
creator {
id
email
displayName
}
}
`;
export const WorkspaceFragmentDoc = gql`
fragment Workspace on workspaces {
id
name
slug
workspaceMembers {
id
user {
id
email
displayName
}
type
}
projects: apps {
...Project
}
}
${ProjectFragmentDoc}`;
export const PrefetchNewAppRegionsFragmentDoc = gql`
fragment PrefetchNewAppRegions on regions {
id
@@ -17218,62 +17481,6 @@ export const RemoteAppGetUsersFragmentDoc = gql`
disabled
}
`;
export const ProjectFragmentDoc = gql`
fragment Project on apps {
id
slug
name
repositoryProductionBranch
subdomain
isProvisioned
createdAt
desiredState
nhostBaseFolder
providersUpdated
config(resolve: true) {
hasura {
adminSecret
}
}
featureFlags {
description
id
name
value
}
appStates(order_by: {createdAt: desc}, limit: 1) {
id
appId
message
stateId
createdAt
}
region {
id
countryCode
awsName
city
}
plan {
id
name
isFree
}
githubRepository {
fullName
}
deployments(limit: 4, order_by: {deploymentEndedAt: desc}) {
id
commitSHA
commitMessage
commitUserName
deploymentStartedAt
deploymentEndedAt
commitUserAvatarUrl
deploymentStatus
}
}
`;
export const GetWorkspaceMembersWorkspaceMemberFragmentDoc = gql`
fragment getWorkspaceMembersWorkspaceMember on workspaceMembers {
id
@@ -17746,6 +17953,49 @@ export type GetRemoteAppRolesQueryResult = Apollo.QueryResult<GetRemoteAppRolesQ
export function refetchGetRemoteAppRolesQuery(variables?: GetRemoteAppRolesQueryVariables) {
return { query: GetRemoteAppRolesDocument, variables: variables }
}
export const GetWorkspaceAndProjectDocument = gql`
query GetWorkspaceAndProject($workspaceSlug: String!, $projectSlug: String) {
workspaces(where: {slug: {_eq: $workspaceSlug}}) {
...Workspace
}
projects: apps(where: {slug: {_eq: $projectSlug}}) {
...Project
}
}
${WorkspaceFragmentDoc}
${ProjectFragmentDoc}`;
/**
* __useGetWorkspaceAndProjectQuery__
*
* To run a query within a React component, call `useGetWorkspaceAndProjectQuery` and pass it any options that fit your needs.
* When your component renders, `useGetWorkspaceAndProjectQuery` returns an object from Apollo Client that contains loading, error, and data properties
* you can use to render your UI.
*
* @param baseOptions options that will be passed into the query, supported options are listed on: https://www.apollographql.com/docs/react/api/react-hooks/#options;
*
* @example
* const { data, loading, error } = useGetWorkspaceAndProjectQuery({
* variables: {
* workspaceSlug: // value for 'workspaceSlug'
* projectSlug: // value for 'projectSlug'
* },
* });
*/
export function useGetWorkspaceAndProjectQuery(baseOptions: Apollo.QueryHookOptions<GetWorkspaceAndProjectQuery, GetWorkspaceAndProjectQueryVariables>) {
const options = {...defaultOptions, ...baseOptions}
return Apollo.useQuery<GetWorkspaceAndProjectQuery, GetWorkspaceAndProjectQueryVariables>(GetWorkspaceAndProjectDocument, options);
}
export function useGetWorkspaceAndProjectLazyQuery(baseOptions?: Apollo.LazyQueryHookOptions<GetWorkspaceAndProjectQuery, GetWorkspaceAndProjectQueryVariables>) {
const options = {...defaultOptions, ...baseOptions}
return Apollo.useLazyQuery<GetWorkspaceAndProjectQuery, GetWorkspaceAndProjectQueryVariables>(GetWorkspaceAndProjectDocument, options);
}
export type GetWorkspaceAndProjectQueryHookResult = ReturnType<typeof useGetWorkspaceAndProjectQuery>;
export type GetWorkspaceAndProjectLazyQueryHookResult = ReturnType<typeof useGetWorkspaceAndProjectLazyQuery>;
export type GetWorkspaceAndProjectQueryResult = Apollo.QueryResult<GetWorkspaceAndProjectQuery, GetWorkspaceAndProjectQueryVariables>;
export function refetchGetWorkspaceAndProjectQuery(variables: GetWorkspaceAndProjectQueryVariables) {
return { query: GetWorkspaceAndProjectDocument, variables: variables }
}
export const InsertApplicationDocument = gql`
mutation insertApplication($app: apps_insert_input!) {
insertApp(object: $app) {
@@ -18282,6 +18532,12 @@ export const GetSignInMethodsDocument = gql`
connection
organization
}
azuread {
enabled
clientId
clientSecret
tenant
}
}
}
}

View File

@@ -1,5 +1,11 @@
# @nhost/docs
## 0.0.15
### Patch Changes
- 0795d1c6: chore: add link to event triggers on the serverless functions page
## 0.0.14
### Patch Changes

View File

@@ -84,6 +84,8 @@ older (`functions/_utils/<utils-files>.js`).
[Environment variables](/platform/environment-variables) are available inside your Serverless Functions. Both in production and when running Nhost locally using the [Nhost CLI](/cli).
The same [environment variables that are used to configure event triggers](https://docs.nhost.io/database/event-triggers#format) can be used to authenticate regular serverless functions.
## Examples
We have multiple examples of Serverless Functions in our [Nhost repository](https://github.com/nhost/nhost/tree/main/examples/serverless-functions/functions).

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/docs",
"version": "0.0.14",
"version": "0.0.15",
"private": true,
"scripts": {
"docusaurus": "docusaurus",

View File

@@ -13,7 +13,7 @@ services:
postgres_password: postgres
postgres_user: postgres
auth:
image: nhost/hasura-auth:0.16.1
image: nhost/hasura-auth:0.16.2
storage:
image: nhost/hasura-storage:0.3.0
auth:

View File

@@ -13,7 +13,7 @@ services:
postgres_password: postgres
postgres_user: postgres
auth:
image: nhost/hasura-auth:0.16.1
image: nhost/hasura-auth:0.16.2
storage:
image: nhost/hasura-storage:0.3.0
auth:

View File

@@ -13,7 +13,7 @@ services:
postgres_password: postgres
postgres_user: postgres
auth:
image: nhost/hasura-auth:0.16.1
image: nhost/hasura-auth:0.16.2
storage:
image: nhost/hasura-storage:0.3.0
auth:

View File

@@ -1,5 +1,19 @@
# @nhost/apollo
## 5.2.1
### Patch Changes
- 0d73e87a: fix(ws): don't open unnecessary connections
- 0d73e87a: fix(ws): increase retry attempts and implement exponential backoff
## 5.2.0
### Patch Changes
- Updated dependencies [a0e093d7]
- @nhost/nhost-js@2.2.0
## 5.1.3
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/apollo",
"version": "5.1.3",
"version": "5.2.1",
"description": "Nhost Apollo Client library",
"license": "MIT",
"keywords": [

View File

@@ -72,7 +72,22 @@ export const createApolloClient = ({
? createRestartableClient({
url: uri.startsWith('https') ? uri.replace(/^https/, 'wss') : uri.replace(/^http/, 'ws'),
shouldRetry: () => true,
retryAttempts: 10,
retryAttempts: 100,
retryWait: async (retries) => {
// start with 1 second delay
const baseDelay = 1000
// max 3 seconds of jitter
const maxJitter = 3000
// exponential backoff with jitter
return new Promise((resolve) =>
setTimeout(
resolve,
baseDelay * Math.pow(2, retries) + Math.floor(Math.random() * maxJitter)
)
)
},
connectionParams: () => ({
headers: {
...headers,
@@ -141,7 +156,7 @@ export const createApolloClient = ({
// update token
token = state.context.accessToken.value
if (!isBrowser) {
if (!isBrowser || !wsClient?.isOpen()) {
return
}

View File

@@ -3,6 +3,7 @@ import { Client, ClientOptions, createClient } from 'graphql-ws'
export interface RestartableClient extends Client {
restart(): void
isOpen(): boolean
}
export function createRestartableClient(options: ClientOptions): RestartableClient {
@@ -10,6 +11,8 @@ export function createRestartableClient(options: ClientOptions): RestartableClie
let restart = () => {
restartRequested = true
}
let connectionOpen = false
let socket: WebSocket
let timedOut: NodeJS.Timeout
@@ -46,6 +49,7 @@ export function createRestartableClient(options: ClientOptions): RestartableClie
opened: (originalSocket) => {
socket = originalSocket as WebSocket
options.on?.opened?.(socket)
connectionOpen = true
restart = () => {
if (socket.readyState === WebSocket.OPEN) {
@@ -63,12 +67,17 @@ export function createRestartableClient(options: ClientOptions): RestartableClie
restartRequested = false
restart()
}
},
closed: (event) => {
options?.on?.closed?.(event)
connectionOpen = false
}
}
})
return {
...client,
restart: () => restart()
restart: () => restart(),
isOpen: () => connectionOpen
}
}

View File

@@ -1,5 +1,20 @@
# @nhost/react-apollo
## 5.0.16
### Patch Changes
- Updated dependencies [0d73e87a]
- Updated dependencies [0d73e87a]
- @nhost/apollo@5.2.1
## 5.0.15
### Patch Changes
- @nhost/apollo@5.2.0
- @nhost/react@2.0.13
## 5.0.14
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/react-apollo",
"version": "5.0.14",
"version": "5.0.16",
"description": "Nhost React Apollo client",
"license": "MIT",
"keywords": [

View File

@@ -1,5 +1,17 @@
# @nhost/react-urql
## 2.0.14
### Patch Changes
- 09cf5d4b: chore(deps): bump `urql` to v4
## 2.0.13
### Patch Changes
- @nhost/react@2.0.13
## 2.0.12
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/react-urql",
"version": "2.0.12",
"version": "2.0.14",
"description": "Nhost React URQL client",
"license": "MIT",
"keywords": [
@@ -75,6 +75,6 @@
"graphql": "16.6.0",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"urql": "^3.0.3"
"urql": "^4.0.0"
}
}

View File

@@ -76,7 +76,7 @@
"husky": "^8.0.1",
"npm-run-all": "^4.1.5",
"prettier": "^2.7.1",
"turbo": "1.8.5",
"turbo": "1.8.6",
"typedoc": "^0.22.18",
"typescript": "4.9.5",
"vite": "^4.0.2",

View File

@@ -1,5 +1,11 @@
# @nhost/hasura-auth-js
## 2.1.0
### Minor Changes
- a0e093d7: fix(exports): don't use conflicting names in exports
## 2.0.2
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/hasura-auth-js",
"version": "2.0.2",
"version": "2.1.0",
"description": "Hasura-auth client",
"license": "MIT",
"keywords": [
@@ -43,7 +43,8 @@
},
"scripts": {
"dev": "vite build --config ./vite.dev.config.js",
"build": "run-p build:lib build:umd",
"build": "run-p typecheck build:lib build:umd",
"typecheck": "tsc --noEmit",
"build:lib": "vite build",
"build:umd": "vite build --config ../../config/vite.lib.umd.config.js",
"test": "vitest run",

View File

@@ -1,4 +1,4 @@
import { ErrorPayload } from './types'
import { AuthErrorPayload } from './types'
export const NETWORK_ERROR_CODE = 0
export const OTHER_ERROR_CODE = 1
@@ -12,8 +12,8 @@ export const STATE_ERROR_CODE = 20
* See https://github.com/statelyai/xstate/issues/3037
*/
export class CodifiedError extends Error {
error: ErrorPayload
constructor(original: Error | ErrorPayload) {
error: AuthErrorPayload
constructor(original: Error | AuthErrorPayload) {
super(original.message)
Error.captureStackTrace(this, this.constructor)
if (original instanceof Error) {
@@ -30,95 +30,95 @@ export class CodifiedError extends Error {
}
}
export type ValidationErrorPayload = ErrorPayload & { status: typeof VALIDATION_ERROR_CODE }
export type ValidationAuthErrorPayload = AuthErrorPayload & { status: typeof VALIDATION_ERROR_CODE }
// TODO share with hasura-auth
export const INVALID_EMAIL_ERROR: ValidationErrorPayload = {
export const INVALID_EMAIL_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-email',
message: 'Email is incorrectly formatted'
}
export const INVALID_MFA_TYPE_ERROR: ValidationErrorPayload = {
export const INVALID_MFA_TYPE_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-mfa-type',
message: 'MFA type is invalid'
}
export const INVALID_MFA_CODE_ERROR: ValidationErrorPayload = {
export const INVALID_MFA_CODE_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-mfa-code',
message: 'MFA code is invalid'
}
export const INVALID_PASSWORD_ERROR: ValidationErrorPayload = {
export const INVALID_PASSWORD_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-password',
message: 'Password is incorrectly formatted'
}
export const INVALID_PHONE_NUMBER_ERROR: ValidationErrorPayload = {
export const INVALID_PHONE_NUMBER_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-phone-number',
message: 'Phone number is incorrectly formatted'
}
export const INVALID_MFA_TICKET_ERROR: ValidationErrorPayload = {
export const INVALID_MFA_TICKET_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-mfa-ticket',
message: 'MFA ticket is invalid'
}
export const NO_MFA_TICKET_ERROR: ValidationErrorPayload = {
export const NO_MFA_TICKET_ERROR: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'no-mfa-ticket',
message: 'No MFA ticket has been provided'
}
export const NO_REFRESH_TOKEN: ValidationErrorPayload = {
export const NO_REFRESH_TOKEN: ValidationAuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'no-refresh-token',
message: 'No refresh token has been provided'
}
export const TOKEN_REFRESHER_RUNNING_ERROR: ErrorPayload = {
export const TOKEN_REFRESHER_RUNNING_ERROR: AuthErrorPayload = {
status: STATE_ERROR_CODE,
error: 'refresher-already-running',
message:
'The token refresher is already running. You must wait until is has finished before submitting a new token.'
}
export const USER_ALREADY_SIGNED_IN: ErrorPayload = {
export const USER_ALREADY_SIGNED_IN: AuthErrorPayload = {
status: STATE_ERROR_CODE,
error: 'already-signed-in',
message: 'User is already signed in'
}
export const USER_UNAUTHENTICATED: ErrorPayload = {
export const USER_UNAUTHENTICATED: AuthErrorPayload = {
status: STATE_ERROR_CODE,
error: 'unauthenticated-user',
message: 'User is not authenticated'
}
export const USER_NOT_ANONYMOUS: ErrorPayload = {
export const USER_NOT_ANONYMOUS: AuthErrorPayload = {
status: STATE_ERROR_CODE,
error: 'user-not-anonymous',
message: 'User is not anonymous'
}
export const EMAIL_NEEDS_VERIFICATION: ErrorPayload = {
export const EMAIL_NEEDS_VERIFICATION: AuthErrorPayload = {
status: STATE_ERROR_CODE,
error: 'unverified-user',
message: 'Email needs verification'
}
export const INVALID_REFRESH_TOKEN = {
export const INVALID_REFRESH_TOKEN: AuthErrorPayload = {
status: VALIDATION_ERROR_CODE,
error: 'invalid-refresh-token',
message: 'Invalid or expired refresh token'
}
export const INVALID_SIGN_IN_METHOD = {
export const INVALID_SIGN_IN_METHOD: AuthErrorPayload = {
status: OTHER_ERROR_CODE,
error: 'invalid-sign-in-method',
message: 'Invalid sign-in method'

View File

@@ -34,13 +34,13 @@ import {
} from './promises'
import {
AuthChangedFunction,
AuthErrorPayload,
ChangeEmailParams,
ChangeEmailResponse,
ChangePasswordParams,
ChangePasswordResponse,
DeanonymizeParams,
DeanonymizeResponse,
ErrorPayload,
JWTClaims,
JWTHasuraClaims,
NhostAuthConstructorParams,
@@ -411,7 +411,7 @@ export class HasuraAuthClient {
*/
async addSecurityKey(
nickname?: string
): Promise<{ error: ErrorPayload | null; key?: SecurityKey }> {
): Promise<{ error: AuthErrorPayload | null; key?: SecurityKey }> {
const { error, key } = await addSecurityKeyPromise(this._client, nickname)
return { error, key }
}

View File

@@ -1,4 +1,4 @@
import { ErrorPayload, User } from '../../types'
import { AuthErrorPayload, User } from '../../types'
export type StateErrorTypes = 'registration' | 'authentication' | 'signout'
@@ -21,7 +21,7 @@ export type AuthContext = {
}
/** Number of times the user tried to get an access token from a refresh token but got a network error */
importTokenAttempts: number
errors: Partial<Record<StateErrorTypes, ErrorPayload>>
errors: Partial<Record<StateErrorTypes, AuthErrorPayload>>
}
export const INITIAL_MACHINE_CONTEXT: AuthContext = {

View File

@@ -24,9 +24,9 @@ import {
} from '../../errors'
import { localStorageGetter, localStorageSetter } from '../../local-storage'
import {
AuthErrorPayload,
AuthOptions,
DeanonymizeResponse,
ErrorPayload,
NhostSession,
NhostSessionResponse,
PasswordlessEmailResponse,
@@ -856,7 +856,7 @@ export const createAuthMachine = ({
error: null
}
}
let error: ErrorPayload | null = null
let error: AuthErrorPayload | null = null
if (autoSignIn) {
const urlToken = getParameterByName('refreshToken') || null
if (urlToken) {
@@ -866,7 +866,7 @@ export const createAuthMachine = ({
})
return { session, error: null }
} catch (exception) {
error = (exception as { error: ErrorPayload }).error
error = (exception as { error: AuthErrorPayload }).error
}
} else {
const error = getParameterByName('error')
@@ -890,7 +890,7 @@ export const createAuthMachine = ({
})
return { session, error: null }
} catch (exception) {
error = (exception as { error: ErrorPayload }).error
error = (exception as { error: AuthErrorPayload }).error
}
}
if (error) {

View File

@@ -1,12 +1,12 @@
import { assign, createMachine, send } from 'xstate'
import { INVALID_EMAIL_ERROR } from '../errors'
import { AuthClient } from '../internal-client'
import { ChangeEmailOptions, ChangeEmailResponse, ErrorPayload } from '../types'
import { AuthErrorPayload, ChangeEmailOptions, ChangeEmailResponse } from '../types'
import { postFetch, rewriteRedirectTo } from '../utils'
import { isValidEmail } from '../utils/validators'
export type ChangeEmailContext = {
error: ErrorPayload | null
error: AuthErrorPayload | null
}
export type ChangeEmailEvents =
@@ -16,7 +16,7 @@ export type ChangeEmailEvents =
options?: ChangeEmailOptions
}
| { type: 'SUCCESS' }
| { type: 'ERROR'; error: ErrorPayload | null }
| { type: 'ERROR'; error: AuthErrorPayload | null }
export type ChangeEmailServices = {
request: { data: ChangeEmailResponse }

View File

@@ -16,9 +16,9 @@ export interface Typegen0 {
}
missingImplementations: {
actions: never
services: never
guards: never
delays: never
guards: never
services: never
}
eventsCausingActions: {
reportError: 'error.platform.requestChange'
@@ -26,13 +26,13 @@ export interface Typegen0 {
saveInvalidEmailError: 'REQUEST'
saveRequestError: 'error.platform.requestChange'
}
eventsCausingServices: {
requestChange: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingGuards: {
invalidEmail: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingServices: {
requestChange: 'REQUEST'
}
matchesStates:
| 'idle'
| 'idle.error'

View File

@@ -1,12 +1,12 @@
import { assign, createMachine, send } from 'xstate'
import { INVALID_PASSWORD_ERROR } from '../errors'
import { AuthClient } from '../internal-client'
import { ChangePasswordResponse, ErrorPayload } from '../types'
import { AuthErrorPayload, ChangePasswordResponse } from '../types'
import { postFetch } from '../utils'
import { isValidPassword } from '../utils/validators'
export type ChangePasswordContext = {
error: ErrorPayload | null
error: AuthErrorPayload | null
}
export type ChangePasswordEvents =
| {
@@ -15,7 +15,7 @@ export type ChangePasswordEvents =
ticket?: string
}
| { type: 'SUCCESS' }
| { type: 'ERROR'; error: ErrorPayload | null }
| { type: 'ERROR'; error: AuthErrorPayload | null }
export type ChangePasswordServices = {
requestChange: { data: ChangePasswordResponse }

View File

@@ -16,9 +16,9 @@ export interface Typegen0 {
}
missingImplementations: {
actions: never
services: never
guards: never
delays: never
guards: never
services: never
}
eventsCausingActions: {
reportError: 'error.platform.requestChange'
@@ -26,13 +26,13 @@ export interface Typegen0 {
saveInvalidPasswordError: 'REQUEST'
saveRequestError: 'error.platform.requestChange'
}
eventsCausingServices: {
requestChange: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingGuards: {
invalidPassword: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingServices: {
requestChange: 'REQUEST'
}
matchesStates:
| 'idle'
| 'idle.error'

View File

@@ -1,11 +1,11 @@
import { assign, createMachine, send } from 'xstate'
import { INVALID_MFA_CODE_ERROR, INVALID_MFA_TYPE_ERROR } from '../errors'
import { AuthClient } from '../internal-client'
import { ErrorPayload } from '../types'
import { AuthErrorPayload } from '../types'
import { getFetch, postFetch } from '../utils'
export type EnableMfaContext = {
error: ErrorPayload | null
error: AuthErrorPayload | null
imageUrl: string | null
secret: string | null
}
@@ -20,9 +20,9 @@ export type EnableMfaEvents =
activeMfaType: 'totp'
}
| { type: 'GENERATED' }
| { type: 'GENERATED_ERROR'; error: ErrorPayload | null }
| { type: 'GENERATED_ERROR'; error: AuthErrorPayload | null }
| { type: 'SUCCESS' }
| { type: 'ERROR'; error: ErrorPayload | null }
| { type: 'ERROR'; error: AuthErrorPayload | null }
export type EnableMfadMachine = ReturnType<typeof createEnableMfaMachine>

View File

@@ -23,9 +23,9 @@ export interface Typegen0 {
}
missingImplementations: {
actions: never
services: never
guards: never
delays: never
guards: never
services: never
}
eventsCausingActions: {
reportError: 'error.platform.activate'
@@ -37,15 +37,15 @@ export interface Typegen0 {
saveInvalidMfaCodeError: 'ACTIVATE'
saveInvalidMfaTypeError: 'ACTIVATE'
}
eventsCausingServices: {
activate: 'ACTIVATE'
generate: 'GENERATE'
}
eventsCausingDelays: {}
eventsCausingGuards: {
invalidMfaCode: 'ACTIVATE'
invalidMfaType: 'ACTIVATE'
}
eventsCausingDelays: {}
eventsCausingServices: {
activate: 'ACTIVATE'
generate: 'GENERATE'
}
matchesStates:
| 'generated'
| 'generated.activated'

View File

@@ -1,12 +1,12 @@
import { assign, createMachine, send } from 'xstate'
import { INVALID_EMAIL_ERROR } from '../errors'
import { AuthClient } from '../internal-client'
import { ErrorPayload, ResetPasswordOptions, ResetPasswordResponse } from '../types'
import { AuthErrorPayload, ResetPasswordOptions, ResetPasswordResponse } from '../types'
import { postFetch, rewriteRedirectTo } from '../utils'
import { isValidEmail } from '../utils/validators'
export type ResetPasswordContext = {
error: ErrorPayload | null
error: AuthErrorPayload | null
}
export type ResetPasswordEvents =
| {
@@ -15,7 +15,7 @@ export type ResetPasswordEvents =
options?: ResetPasswordOptions
}
| { type: 'SUCCESS' }
| { type: 'ERROR'; error: ErrorPayload | null }
| { type: 'ERROR'; error: AuthErrorPayload | null }
export type ResetPasswordServices = {
requestChange: { data: ResetPasswordResponse }

View File

@@ -16,9 +16,9 @@ export interface Typegen0 {
}
missingImplementations: {
actions: never
services: never
guards: never
delays: never
guards: never
services: never
}
eventsCausingActions: {
reportError: 'error.platform.requestChange'
@@ -26,13 +26,13 @@ export interface Typegen0 {
saveInvalidEmailError: 'REQUEST'
saveRequestError: 'error.platform.requestChange'
}
eventsCausingServices: {
requestChange: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingGuards: {
invalidEmail: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingServices: {
requestChange: 'REQUEST'
}
matchesStates:
| 'idle'
| 'idle.error'

View File

@@ -1,12 +1,16 @@
import { assign, createMachine, send } from 'xstate'
import { INVALID_EMAIL_ERROR } from '../errors'
import { AuthClient } from '../internal-client'
import { ErrorPayload, SendVerificationEmailOptions, SendVerificationEmailResponse } from '../types'
import {
AuthErrorPayload,
SendVerificationEmailOptions,
SendVerificationEmailResponse
} from '../types'
import { postFetch, rewriteRedirectTo } from '../utils'
import { isValidEmail } from '../utils/validators'
export type SendVerificationEmailContext = {
error: ErrorPayload | null
error: AuthErrorPayload | null
}
export type SendVerificationEmailEvents =
@@ -16,7 +20,7 @@ export type SendVerificationEmailEvents =
options?: SendVerificationEmailOptions
}
| { type: 'SUCCESS' }
| { type: 'ERROR'; error: ErrorPayload | null }
| { type: 'ERROR'; error: AuthErrorPayload | null }
export type SendVerificationEmailServices = {
request: { data: SendVerificationEmailResponse }

View File

@@ -16,9 +16,9 @@ export interface Typegen0 {
}
missingImplementations: {
actions: never
services: never
guards: never
delays: never
guards: never
services: never
}
eventsCausingActions: {
reportError: 'error.platform.request'
@@ -26,13 +26,13 @@ export interface Typegen0 {
saveInvalidEmailError: 'REQUEST'
saveRequestError: 'error.platform.request'
}
eventsCausingServices: {
request: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingGuards: {
invalidEmail: 'REQUEST'
}
eventsCausingDelays: {}
eventsCausingServices: {
request: 'REQUEST'
}
matchesStates:
| 'idle'
| 'idle.error'

View File

@@ -6,14 +6,14 @@ import {
import { postFetch } from '..'
import { CodifiedError } from '../errors'
import { AuthClient } from '../internal-client'
import { ErrorPayload, SecurityKey } from '../types'
import { ActionErrorState, ActionLoadingState, ActionSuccessState } from './types'
import { AuthErrorPayload, SecurityKey } from '../types'
import { AuthActionErrorState, AuthActionLoadingState, AuthActionSuccessState } from './types'
export interface AddSecurityKeyHandlerResult extends ActionErrorState, ActionSuccessState {
export interface AddSecurityKeyHandlerResult extends AuthActionErrorState, AuthActionSuccessState {
key?: SecurityKey
}
export interface AddSecurityKeyState extends AddSecurityKeyHandlerResult, ActionLoadingState {}
export interface AddSecurityKeyState extends AddSecurityKeyHandlerResult, AuthActionLoadingState {}
export const addSecurityKeyPromise = async (
{ backendUrl, interpreter }: AuthClient,
@@ -38,7 +38,7 @@ export const addSecurityKeyPromise = async (
)
return { key, isError: false, error: null, isSuccess: true }
} catch (e) {
const { error } = e as { error: ErrorPayload }
const { error } = e as { error: AuthErrorPayload }
return { isError: true, error, isSuccess: false }
}
}

View File

@@ -3,10 +3,12 @@ import { InterpreterFrom } from 'xstate'
import { ChangeEmailMachine } from '../machines'
import { ChangeEmailOptions } from '../types'
import { ActionErrorState, ActionLoadingState, NeedsEmailVerificationState } from './types'
export interface ChangeEmailHandlerResult extends ActionErrorState, NeedsEmailVerificationState {}
import { AuthActionErrorState, AuthActionLoadingState, NeedsEmailVerificationState } from './types'
export interface ChangeEmailHandlerResult
extends AuthActionErrorState,
NeedsEmailVerificationState {}
export interface ChangeEmailState extends ChangeEmailHandlerResult, ActionLoadingState {}
export interface ChangeEmailState extends ChangeEmailHandlerResult, AuthActionLoadingState {}
export const changeEmailPromise = async (
interpreter: InterpreterFrom<ChangeEmailMachine>,

View File

@@ -2,11 +2,11 @@ import { InterpreterFrom } from 'xstate'
import { ChangePasswordMachine } from '../machines'
import { ActionErrorState, ActionLoadingState, ActionSuccessState } from './types'
import { AuthActionErrorState, AuthActionLoadingState, AuthActionSuccessState } from './types'
export interface ChangePasswordState extends ChangePasswordHandlerResult, ActionLoadingState {}
export interface ChangePasswordState extends ChangePasswordHandlerResult, AuthActionLoadingState {}
export interface ChangePasswordHandlerResult extends ActionErrorState, ActionSuccessState {}
export interface ChangePasswordHandlerResult extends AuthActionErrorState, AuthActionSuccessState {}
export const changePasswordPromise = async (
interpreter: InterpreterFrom<ChangePasswordMachine>,

View File

@@ -2,9 +2,9 @@ import { InterpreterFrom } from 'xstate'
import { EnableMfadMachine } from '../machines'
import { ActionErrorState } from './types'
import { AuthActionErrorState } from './types'
export interface GenerateQrCodeHandlerResult extends ActionErrorState {
export interface GenerateQrCodeHandlerResult extends AuthActionErrorState {
qrCodeDataUrl: string
isGenerated: boolean
}
@@ -12,7 +12,7 @@ export interface GenerateQrCodeHandlerResult extends ActionErrorState {
export interface GenerateQrCodeState extends GenerateQrCodeHandlerResult {
isGenerating: boolean
}
export interface ActivateMfaHandlerResult extends ActionErrorState {
export interface ActivateMfaHandlerResult extends AuthActionErrorState {
isActivated: boolean
}

View File

@@ -3,14 +3,14 @@ import { InterpreterFrom } from 'xstate'
import { ResetPasswordMachine } from '../machines'
import { ResetPasswordOptions } from '../types'
import { ActionErrorState, ActionLoadingState } from './types'
import { AuthActionErrorState, AuthActionLoadingState } from './types'
export interface ResetPasswordHandlerResult extends ActionErrorState {
export interface ResetPasswordHandlerResult extends AuthActionErrorState {
/** Returns `true` when an email to reset the password has been sent */
isSent: boolean
}
export interface ResetPasswordState extends ResetPasswordHandlerResult, ActionLoadingState {}
export interface ResetPasswordState extends ResetPasswordHandlerResult, AuthActionLoadingState {}
export const resetPasswordPromise = async (
interpreter: InterpreterFrom<ResetPasswordMachine>,

View File

@@ -3,15 +3,15 @@ import { InterpreterFrom } from 'xstate'
import { SendVerificationEmailMachine } from '../machines'
import { SendVerificationEmailOptions } from '../types'
import { ActionErrorState, ActionLoadingState } from './types'
import { AuthActionErrorState, AuthActionLoadingState } from './types'
export interface SendVerificationEmailHandlerResult extends ActionErrorState {
export interface SendVerificationEmailHandlerResult extends AuthActionErrorState {
/** Returns `true` when a new verification email has been sent */
isSent: boolean
}
export interface SendVerificationEmailState
extends ActionLoadingState,
extends AuthActionLoadingState,
SendVerificationEmailHandlerResult {}
export const sendVerificationEmailPromise = (

View File

@@ -1,10 +1,12 @@
import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import { ActionLoadingState, SessionActionHandlerResult } from './types'
import { AuthActionLoadingState, SessionActionHandlerResult } from './types'
export interface SignInAnonymousHandlerResult extends SessionActionHandlerResult {}
export interface SignInAnonymousState extends SignInAnonymousHandlerResult, ActionLoadingState {}
export interface SignInAnonymousState
extends SignInAnonymousHandlerResult,
AuthActionLoadingState {}
export const signInAnonymousPromise = (
interpreter: AuthInterpreter

View File

@@ -2,7 +2,7 @@ import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import {
ActionLoadingState,
AuthActionLoadingState,
NeedsEmailVerificationState,
SessionActionHandlerResult
} from './types'
@@ -18,7 +18,7 @@ export interface SignInEmailPasswordHandlerResult
export interface SignInEmailPasswordState
extends SignInEmailPasswordHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export const signInEmailPasswordPromise = (
interpreter: AuthInterpreter,

View File

@@ -2,13 +2,13 @@ import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import { PasswordlessOptions } from '../types'
import { ActionErrorState, ActionLoadingState, ActionSuccessState } from './types'
import { AuthActionErrorState, AuthActionLoadingState, AuthActionSuccessState } from './types'
export interface SignInEmailPasswordlessHandlerResult
extends ActionErrorState,
ActionSuccessState {}
extends AuthActionErrorState,
AuthActionSuccessState {}
export interface SignInEmailPasswordlessState
extends SignInEmailPasswordlessHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export const signInEmailPasswordlessPromise = (
interpreter: AuthInterpreter,

View File

@@ -2,7 +2,7 @@ import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import {
ActionLoadingState,
AuthActionLoadingState,
NeedsEmailVerificationState,
SessionActionHandlerResult
} from './types'
@@ -13,7 +13,7 @@ export interface SignInSecurityKeyPasswordlessHandlerResult
export interface SignInSecurityKeyPasswordlessState
extends SignInSecurityKeyPasswordlessHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export const signInEmailSecurityKeyPromise = (interpreter: AuthInterpreter, email: string) =>
new Promise<SignInSecurityKeyPasswordlessHandlerResult>((resolve) => {

View File

@@ -1,11 +1,11 @@
import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import { ActionLoadingState, SessionActionHandlerResult } from './types'
import { AuthActionLoadingState, SessionActionHandlerResult } from './types'
export interface SignInMfaTotpHandlerResult extends SessionActionHandlerResult {}
export interface SignInMfaTotpState extends SignInMfaTotpHandlerResult, ActionLoadingState {}
export interface SignInMfaTotpState extends SignInMfaTotpHandlerResult, AuthActionLoadingState {}
export const signInMfaTotpPromise = (interpreter: AuthInterpreter, otp: string, ticket?: string) =>
new Promise<SignInMfaTotpHandlerResult>((resolve) => {

View File

@@ -2,13 +2,15 @@ import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import { PasswordlessOptions } from '../types'
import { ActionErrorState, ActionLoadingState, ActionSuccessState } from './types'
import { AuthActionErrorState, AuthActionLoadingState, AuthActionSuccessState } from './types'
export interface SignInSmsPasswordlessState
extends SignInSmsPasswordlessHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export interface SignInSmsPasswordlessHandlerResult extends ActionErrorState, ActionSuccessState {
export interface SignInSmsPasswordlessHandlerResult
extends AuthActionErrorState,
AuthActionSuccessState {
/**
* Returns true when the one-time password has been sent over by SMS, and the user needs to send it back to complete sign-in.
*/

View File

@@ -1,12 +1,12 @@
import { USER_ALREADY_SIGNED_IN } from '../errors'
import { AuthInterpreter } from '../machines'
import { ActionLoadingState, SessionActionHandlerResult } from './types'
import { AuthActionLoadingState, SessionActionHandlerResult } from './types'
export interface SignInSmsPasswordlessOtpHandlerResult extends SessionActionHandlerResult {}
export interface SignInSmsPasswordlessOtpState
extends SignInSmsPasswordlessOtpHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export const signInSmsPasswordlessOtpPromise = (
interpreter: AuthInterpreter,

View File

@@ -1,10 +1,10 @@
import { USER_UNAUTHENTICATED } from '../errors'
import { AuthInterpreter } from '../machines'
import { ActionErrorState, ActionLoadingState, ActionSuccessState } from './types'
import { AuthActionErrorState, AuthActionLoadingState, AuthActionSuccessState } from './types'
export interface SignOutlessHandlerResult extends ActionErrorState, ActionSuccessState {}
export interface SignOutlessState extends SignOutlessHandlerResult, ActionLoadingState {}
export interface SignOutlessHandlerResult extends AuthActionErrorState, AuthActionSuccessState {}
export interface SignOutlessState extends SignOutlessHandlerResult, AuthActionLoadingState {}
export const signOutPromise = async (
interpreter: AuthInterpreter,

View File

@@ -3,7 +3,7 @@ import { AuthInterpreter } from '../machines'
import { SignUpOptions } from '../types'
import {
ActionLoadingState,
AuthActionLoadingState,
NeedsEmailVerificationState,
SessionActionHandlerResult
} from './types'
@@ -14,7 +14,7 @@ export interface SignUpEmailPasswordHandlerResult
export interface SignUpEmailPasswordState
extends SignUpEmailPasswordHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export const signUpEmailPasswordPromise = (
interpreter: AuthInterpreter,

View File

@@ -3,7 +3,7 @@ import { AuthInterpreter } from '../machines'
import { SignUpSecurityKeyOptions } from '../types'
import {
ActionLoadingState,
AuthActionLoadingState,
NeedsEmailVerificationState,
SessionActionHandlerResult
} from './types'
@@ -14,7 +14,7 @@ export interface SignUpSecurityKeyHandlerResult
export interface SignUpSecurityKeyState
extends SignUpSecurityKeyHandlerResult,
ActionLoadingState {}
AuthActionLoadingState {}
export const signUpEmailSecurityKeyPromise = (
interpreter: AuthInterpreter,

View File

@@ -1,28 +1,28 @@
import { ErrorPayload, User } from '../types'
import { AuthErrorPayload, User } from '../types'
export interface ActionErrorState {
export interface AuthActionErrorState {
/**
* @return `true` if an error occurred
* @depreacted use `!isSuccess` or `!!error` instead
* */
isError: boolean
/** Provides details about the error */
error: ErrorPayload | null
error: AuthErrorPayload | null
}
export interface ActionLoadingState {
export interface AuthActionLoadingState {
/**
* @return `true` when the action is executing, `false` when it finished its execution.
*/
isLoading: boolean
}
export interface ActionSuccessState {
export interface AuthActionSuccessState {
/** Returns `true` if the action is successful. */
isSuccess: boolean
}
export interface SessionActionHandlerResult extends ActionSuccessState, ActionErrorState {
export interface SessionActionHandlerResult extends AuthActionSuccessState, AuthActionErrorState {
/** User information */
user: User | null
/** Access token (JWT) */

View File

@@ -1,5 +1,5 @@
// TODO shared with other packages
export type ErrorPayload = {
export type AuthErrorPayload = {
error: string
status: number
message: string

View File

@@ -1,13 +1,13 @@
import { ErrorPayload, NhostSession } from './common'
import { AuthErrorPayload, NhostSession } from './common'
// Hasura-auth API response types
export interface NullableErrorResponse {
error: ErrorPayload | null
error: AuthErrorPayload | null
}
/** session payload from common hasura-auth responses */
export type NhostSessionResponse =
| { session: null; error: ErrorPayload }
| { session: null; error: AuthErrorPayload }
| { session: NhostSession | null; error: null }
/** payload from hasura-auth endpoint /signin/email-password */
@@ -16,7 +16,7 @@ export interface SignInResponse {
mfa: {
ticket: string
} | null
error: ErrorPayload | null
error: AuthErrorPayload | null
}
/** payload from hasura-auth endpoint /signup/email-password */

View File

@@ -1,5 +1,15 @@
# @nhost/hasura-storage-js
## 2.1.0
### Minor Changes
- a0e093d7: fix(exports): don't use conflicting names in exports
### Patch Changes
- bb8803a1: fix(presigned-url): don't alter URL when no transformation params were provided
## 2.0.5
### Patch Changes

View File

@@ -1,6 +1,6 @@
{
"name": "@nhost/hasura-storage-js",
"version": "2.0.5",
"version": "2.1.0",
"description": "Hasura-storage client",
"license": "MIT",
"keywords": [
@@ -41,7 +41,8 @@
},
"scripts": {
"dev": "vite build --config ../../config/vite.lib.dev.config.js",
"build": "run-p build:lib build:umd",
"build": "run-p typecheck build:lib build:umd",
"typecheck": "tsc --noEmit",
"build:lib": "vite build",
"build:umd": "vite build --config ../../config/vite.lib.umd.config.js",
"e2e": "start-test e2e:backend http-get://localhost:9695 ci:test",

View File

@@ -1,12 +1,12 @@
import FormData from 'form-data'
import { assign, createMachine } from 'xstate'
import { ErrorPayload, FileUploadConfig } from '../utils'
import { FileUploadConfig, StorageErrorPayload } from '../utils'
import { fetchUpload } from '../utils/upload'
export type FileUploadContext = {
progress: number | null
loaded: number
error: ErrorPayload | null
error: StorageErrorPayload | null
id?: string
bucketId?: string
file?: File
@@ -24,7 +24,7 @@ export type FileUploadEvents =
} & FileUploadConfig)
| { type: 'UPLOAD_PROGRESS'; progress: number; loaded: number; additions: number }
| { type: 'UPLOAD_DONE'; id: string; bucketId: string }
| { type: 'UPLOAD_ERROR'; error: ErrorPayload }
| { type: 'UPLOAD_ERROR'; error: StorageErrorPayload }
| { type: 'CANCEL' }
| { type: 'DESTROY' }

View File

@@ -1,7 +1,7 @@
import { InterpreterFrom } from 'xstate'
import { FileItemRef, FileUploadMachine } from '../machines'
import { ActionErrorState, FileUploadConfig, StorageUploadFileParams } from '../utils'
import { FileUploadConfig, StorageActionErrorState, StorageUploadFileParams } from '../utils'
export interface UploadProgressState {
/**
@@ -14,7 +14,7 @@ export interface UploadProgressState {
progress: number | null
}
export interface UploadFileHandlerResult extends ActionErrorState {
export interface UploadFileHandlerResult extends StorageActionErrorState {
/**
* Returns `true` when the file has been successfully uploaded.
*/

View File

@@ -3,7 +3,7 @@ import appendImageTransformationParameters from './appendImageTransformationPara
test('should append image transformation parameters to a simple URL', () => {
expect(
appendImageTransformationParameters('https://example.com/', {
appendImageTransformationParameters('https://example.com', {
width: 100,
height: 100,
blur: 50,
@@ -25,7 +25,7 @@ test('should append image transformation parameters to a URL with existing query
test('should not append falsy values', () => {
expect(
appendImageTransformationParameters('https://example.com/', {
appendImageTransformationParameters('https://example.com', {
width: undefined,
height: 100,
blur: undefined,
@@ -35,7 +35,9 @@ test('should not append falsy values', () => {
})
test('should keep the original URL if no transformation parameters are provided', () => {
expect(appendImageTransformationParameters('https://example.com/', {})).toBe(
'https://example.com/'
)
expect(appendImageTransformationParameters('https://example.com')).toBe('https://example.com')
expect(
appendImageTransformationParameters('https://example.com/?param1=test_data&param2=test_data')
).toBe('https://example.com/?param1=test_data&param2=test_data')
expect(appendImageTransformationParameters('https://example.com/')).toBe('https://example.com/')
})

Some files were not shown because too many files have changed in this diff Show More