0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-03-31 22:51:25 -05:00

Merge pull request #2245 from logto-io/merge/sie-v2-102401

chore: merge master into `feature/sie-v2`
This commit is contained in:
Xiao Yijun 2022-10-24 15:07:05 +08:00 committed by GitHub
commit 569069524e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
464 changed files with 2229 additions and 1076 deletions

36
.changeset/README.md Normal file
View file

@ -0,0 +1,36 @@
# Changesets
Hello and welcome! This folder has been automatically generated by `@changesets/cli`, a build tool that works
with multi-package repos, or single-package repos to help you version and publish your code. You can
find the full documentation for it [in our repository](https://github.com/changesets/changesets)
We have a quick list of common questions to get you started engaging with this project in
[our documentation](https://github.com/changesets/changesets/blob/main/docs/common-questions.md)
## Logto publish flow
For now, Changesets only supports tagging packages separately, instead of tagging a "release group". There is an [open issue](https://github.com/changesets/changesets/issues/683) that is still hanging in the air.
So, we are using our own "grouping" release strategy in this monorepo:
### Core
The release group that includes the Logto core products, which consists of the following packages:
- @logto/console
- @logto/core (main)
- @logto/integration-tests
- @logto/ui
Their version will be in sync, and forms our main release.
### CLI
The release group that includes Logto CLI and its aliases:
- @logto/cli (main)
- @logto/create
### Others
For simplicity, we will tag other **public** packages separately and publish them to NPM. But in most cases, no GitHub release will present for these packages.

31
.changeset/config.json Normal file
View file

@ -0,0 +1,31 @@
{
"$schema": "https://unpkg.com/@changesets/config@2.2.0/schema.json",
"changelog": "@changesets/cli/changelog",
"commit": false,
"//": "CAUTION: When updating the fields below, you should also update README accordingly.",
"fixed": [[
"@logto/cli",
"@logto/create"
], [
"@logto/console",
"@logto/core",
"@logto/integration-tests",
"@logto/ui"
]],
"//": "Ignore other release group members, only keep the major one.",
"ignore": [
"@logto/create",
"@logto/console",
"@logto/integration-tests",
"@logto/ui"
],
"linked": [[
"@logto/phrases",
"@logto/phrases-ui",
"@logto/schemas"
]],
"//": "END OF CAUTION",
"access": "restricted",
"baseBranch": "master",
"updateInternalDependencies": "patch"
}

1
.github/CODEOWNERS vendored
View file

@ -1 +1,2 @@
/packages/schemas/tables @simeng-li @wangsijie
/.changeset @gao-sun

View file

@ -78,6 +78,42 @@ jobs:
az config set extension.use_dynamic_install=yes_without_prompt
az containerapp update -n logto-dev -g LogtoDev --image svhd/logto:edge
release-changesets:
runs-on: ubuntu-latest
if: ${{ !startsWith(github.ref, 'refs/tags/') }}
steps:
- uses: actions/checkout@v3
- name: Setup Node and pnpm
uses: silverhand-io/actions-node-pnpm-run-steps@v2
- name: Import GPG key
uses: crazy-max/ghaction-import-gpg@v5
with:
gpg_private_key: ${{ secrets.BOT_GPG_KEY }}
passphrase: ${{ secrets.BOT_GPG_PASSPHRASE }}
git_user_signingkey: true
git_commit_gpgsign: true
git_tag_gpgsign: true
- name: Configure Git user
run: |
git config --global user.email bot@silverhand.io
git config --global user.name silverhand-bot
- name: Create release pull request
uses: changesets/action@v1
with:
# Disable temporarily since https://github.com/changesets/changesets/issues/833.
# Wait for our customized publish flow.
# publish: pnpm changeset tag
commit: 'release: version packages'
title: 'release: version packages'
# Create by our rules defined in `/.changeset/README.md`.
createGithubReleases: false
setupGitUser: false
env:
GITHUB_TOKEN: ${{ secrets.BOT_PAT }}
create-github-release:
environment: release

View file

@ -4,7 +4,6 @@
"license": "MPL-2.0",
"scripts": {
"preinstall": "npx only-allow pnpm",
"version": "pnpm i --frozen-lockfile=false && git add pnpm-lock.yaml",
"prepare": "if test \"$NODE_ENV\" != \"production\" && test \"$CI\" != \"true\" ; then husky install ; fi",
"prepack": "pnpm -r prepack",
"dev": "pnpm -r prepack && pnpm start:dev",
@ -18,6 +17,7 @@
"ci:test": "pnpm -r --parallel test:ci"
},
"devDependencies": {
"@changesets/cli": "^2.25.0",
"@commitlint/cli": "^17.0.0",
"@commitlint/config-conventional": "^17.0.0",
"@commitlint/types": "^17.0.0",

View file

@ -40,8 +40,8 @@
"url": "https://github.com/logto-io/logto/issues"
},
"dependencies": {
"@logto/schemas": "^1.0.0-beta.12",
"@logto/shared": "^1.0.0-beta.12",
"@logto/schemas": "workspace:^",
"@logto/shared": "workspace:^",
"@silverhand/essentials": "^1.3.0",
"chalk": "^4.1.2",
"decamelize": "^5.0.0",
@ -60,10 +60,10 @@
"slonik-sql-tag-raw": "^1.1.4",
"tar": "^6.1.11",
"yargs": "^17.6.0",
"zod": "^3.18.0"
"zod": "^3.19.1"
},
"devDependencies": {
"@silverhand/eslint-config": "1.2.0",
"@silverhand/eslint-config": "1.3.0",
"@silverhand/jest-config": "1.2.2",
"@silverhand/ts-config": "1.2.1",
"@types/fs-extra": "^9.0.13",

View file

@ -1,4 +1,4 @@
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import { log } from '../../utilities';
import { addConnectors, addOfficialConnectors, inquireInstancePath } from './utils';

View file

@ -1,5 +1,5 @@
import { noop } from '@silverhand/essentials';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import add from './add';
import list from './list';

View file

@ -1,5 +1,5 @@
import chalk from 'chalk';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import { getConnectorPackagesFrom, isOfficialConnector } from './utils';

View file

@ -1,6 +1,6 @@
import chalk from 'chalk';
import fsExtra from 'fs-extra';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import { log } from '../../utilities';
import { getConnectorPackagesFrom } from './utils';

View file

@ -2,7 +2,7 @@ import { createMockPool } from 'slonik';
import * as functions from '.';
import * as queries from '../../../queries/logto-config';
import { QueryType } from '../../../test-utilities';
import type { QueryType } from '../../../test-utilities';
import { chooseAlterationsByVersion } from './version';
const mockQuery: jest.MockedFunction<QueryType> = jest.fn();

View file

@ -1,12 +1,12 @@
import path from 'path';
import { AlterationScript } from '@logto/schemas/lib/types/alteration';
import type { AlterationScript } from '@logto/schemas/lib/types/alteration';
import { findPackage } from '@logto/shared';
import { conditionalString } from '@silverhand/essentials';
import chalk from 'chalk';
import { copy, existsSync, remove, readdir } from 'fs-extra';
import { DatabasePool } from 'slonik';
import { CommandModule } from 'yargs';
import type { DatabasePool } from 'slonik';
import type { CommandModule } from 'yargs';
import { createPoolFromConfig } from '../../../database';
import {
@ -14,7 +14,7 @@ import {
updateDatabaseTimestamp,
} from '../../../queries/logto-config';
import { getPathInModule, log } from '../../../utilities';
import { AlterationFile } from './type';
import type { AlterationFile } from './type';
import { chooseAlterationsByVersion } from './version';
const alterationFilenameRegex = /-(\d+)-?.*\.js$/;

View file

@ -4,7 +4,7 @@ import inquirer from 'inquirer';
import { SemVer, compare, eq, gt } from 'semver';
import { findLastIndex, isTty, log } from '../../../utilities';
import { AlterationFile } from './type';
import type { AlterationFile } from './type';
const getVersionFromFilename = (filename: string) => {
try {

View file

@ -1,7 +1,8 @@
import { logtoConfigGuards, LogtoConfigKey, logtoConfigKeys } from '@logto/schemas';
import type { LogtoConfigKey } from '@logto/schemas';
import { logtoConfigGuards, logtoConfigKeys } from '@logto/schemas';
import { deduplicate, noop } from '@silverhand/essentials';
import chalk from 'chalk';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import { createPoolFromConfig } from '../../database';
import { getRowsByKeys, updateValueByKey } from '../../queries/logto-config';

View file

@ -1,5 +1,5 @@
import { noop } from '@silverhand/essentials';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import alteration from './alteration';
import config from './config';

View file

@ -4,9 +4,10 @@ import path from 'path';
import { logtoConfigGuards, LogtoOidcConfigKey, seeds } from '@logto/schemas';
import { buildApplicationSecret } from '@logto/shared';
import chalk from 'chalk';
import { DatabasePool, DatabaseTransactionConnection, sql } from 'slonik';
import type { DatabasePool, DatabaseTransactionConnection } from 'slonik';
import { sql } from 'slonik';
import { raw } from 'slonik-sql-tag-raw';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import { z } from 'zod';
import { createPoolAndDatabaseIfNeeded, insertInto } from '../../../database';

View file

@ -2,7 +2,8 @@ import { generateKeyPair } from 'crypto';
import { readFile } from 'fs/promises';
import { promisify } from 'util';
import { LogtoOidcConfigKey, LogtoOidcConfigType } from '@logto/schemas';
import type { LogtoOidcConfigType } from '@logto/schemas';
import { LogtoOidcConfigKey } from '@logto/schemas';
import { getEnv, getEnvAsStringArray } from '@silverhand/essentials';
import { nanoid } from 'nanoid';

View file

@ -1,5 +1,5 @@
import chalk from 'chalk';
import { CommandModule } from 'yargs';
import type { CommandModule } from 'yargs';
import { getDatabaseUrlFromConfig } from '../../database';
import { log } from '../../utilities';

View file

@ -1,4 +1,4 @@
import { SchemaLike } from '@logto/schemas';
import type { SchemaLike } from '@logto/schemas';
import { convertToPrimitiveOrSql } from '@logto/shared';
import { assert } from '@silverhand/essentials';
import decamelize from 'decamelize';

View file

@ -1,5 +1,5 @@
declare module 'slonik-interceptor-preset' {
import { Interceptor } from 'slonik';
import type { Interceptor } from 'slonik';
export const createInterceptors: (config?: {
benchmarkQueries: boolean;

View file

@ -2,7 +2,8 @@ import { AlterationStateKey, LogtoConfigs } from '@logto/schemas';
import { convertToIdentifiers } from '@logto/shared';
import { createMockPool, createMockQueryResult, sql } from 'slonik';
import { expectSqlAssert, QueryType } from '../test-utilities';
import type { QueryType } from '../test-utilities';
import { expectSqlAssert } from '../test-utilities';
import { updateDatabaseTimestamp, getCurrentDatabaseAlterationTimestamp } from './logto-config';
const mockQuery: jest.MockedFunction<QueryType> = jest.fn();

View file

@ -1,14 +1,9 @@
import {
AlterationState,
LogtoConfig,
logtoConfigGuards,
LogtoConfigKey,
LogtoConfigs,
AlterationStateKey,
} from '@logto/schemas';
import type { AlterationState, LogtoConfig, LogtoConfigKey } from '@logto/schemas';
import { logtoConfigGuards, LogtoConfigs, AlterationStateKey } from '@logto/schemas';
import { convertToIdentifiers } from '@logto/shared';
import { Nullable } from '@silverhand/essentials';
import { DatabasePool, DatabaseTransactionConnection, sql } from 'slonik';
import type { Nullable } from '@silverhand/essentials';
import type { DatabasePool, DatabaseTransactionConnection } from 'slonik';
import { sql } from 'slonik';
import { z } from 'zod';
const { table, fields } = convertToIdentifiers(LogtoConfigs);

View file

@ -1,7 +1,7 @@
// Copied from core
import { QueryResult, QueryResultRow } from 'slonik';
import { PrimitiveValueExpression } from 'slonik/dist/src/types.d';
import type { QueryResult, QueryResultRow } from 'slonik';
import type { PrimitiveValueExpression } from 'slonik/dist/src/types.d';
export type QueryType = (
sql: string,

View file

@ -2,9 +2,11 @@ import { execSync } from 'child_process';
import { createWriteStream } from 'fs';
import path from 'path';
import { conditionalString, Optional } from '@silverhand/essentials';
import type { Optional } from '@silverhand/essentials';
import { conditionalString } from '@silverhand/essentials';
import chalk from 'chalk';
import got, { Progress } from 'got';
import type { Progress } from 'got';
import got from 'got';
import { HttpsProxyAgent } from 'hpagent';
import inquirer from 'inquirer';
import ora from 'ora';

View file

@ -18,19 +18,19 @@
},
"devDependencies": {
"@fontsource/roboto-mono": "^4.5.7",
"@logto/core-kit": "1.0.0-beta.18",
"@logto/language-kit": "1.0.0-beta.19",
"@logto/phrases": "^1.0.0-beta.12",
"@logto/phrases-ui": "^1.0.0-beta.12",
"@logto/react": "1.0.0-beta.8",
"@logto/schemas": "^1.0.0-beta.12",
"@logto/core-kit": "1.0.0-beta.20",
"@logto/language-kit": "1.0.0-beta.20",
"@logto/phrases": "workspace:^",
"@logto/phrases-ui": "workspace:^",
"@logto/react": "1.0.0-beta.10",
"@logto/schemas": "workspace:^",
"@mdx-js/react": "^1.6.22",
"@parcel/core": "2.7.0",
"@parcel/transformer-mdx": "2.7.0",
"@parcel/transformer-sass": "2.7.0",
"@parcel/transformer-svg-react": "2.7.0",
"@silverhand/eslint-config": "1.2.0",
"@silverhand/eslint-config-react": "1.2.1",
"@silverhand/eslint-config": "1.3.0",
"@silverhand/eslint-config-react": "1.3.0",
"@silverhand/essentials": "^1.3.0",
"@silverhand/ts-config": "1.2.1",
"@silverhand/ts-config-react": "1.2.1",
@ -83,7 +83,7 @@
"stylelint": "^14.9.1",
"swr": "^1.3.0",
"typescript": "^4.7.4",
"zod": "^3.18.0"
"zod": "^3.19.1"
},
"alias": {
"@/*": "./src/$1",

View file

@ -1,6 +1,7 @@
import { forwardRef } from 'react';
import Button, { Props as ButtonProps } from '../Button';
import type { Props as ButtonProps } from '../Button';
import Button from '../Button';
const ActionMenuButton = forwardRef<HTMLDivElement, ButtonProps>((props, anchorReference) => (
<div ref={anchorReference} style={{ display: 'inline-block' }}>

View file

@ -1,6 +1,7 @@
import { ReactNode, useRef, useState } from 'react';
import type { ReactNode } from 'react';
import { useRef, useState } from 'react';
import { Props as ButtonProps } from '../Button';
import type { Props as ButtonProps } from '../Button';
import Dropdown from '../Dropdown';
import ActionMenuButton from './ActionMenuButton';
import * as styles from './index.module.scss';

View file

@ -1,6 +1,6 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
import Info from '@/assets/images/info.svg';
import LinkButton from '@/components/LinkButton';

View file

@ -1,5 +1,6 @@
import { AppearanceMode } from '@logto/schemas';
import { ReactNode, useEffect } from 'react';
import type { ReactNode } from 'react';
import { useEffect } from 'react';
import useUserPreferences from '@/hooks/use-user-preferences';
import initI18n from '@/i18n/init';

View file

@ -1,4 +1,4 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import { AppearanceMode } from '@logto/schemas';
import DiscordDark from '@/assets/images/discord-dark.svg';

View file

@ -1,6 +1,8 @@
import classNames from 'classnames';
import { ReactChild, ReactNode, useMemo, useState } from 'react';
import { TFuncKey, useTranslation } from 'react-i18next';
import type { ReactChild, ReactNode } from 'react';
import { useMemo, useState } from 'react';
import type { TFuncKey } from 'react-i18next';
import { useTranslation } from 'react-i18next';
import { Link } from 'react-router-dom';
import { getPath } from '../../utils';

View file

@ -1,4 +1,4 @@
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,6 +1,6 @@
import { Optional } from '@silverhand/essentials';
import { FC, ReactNode } from 'react';
import { TFuncKey } from 'react-i18next';
import type { Optional } from '@silverhand/essentials';
import type { FC, ReactNode } from 'react';
import type { TFuncKey } from 'react-i18next';
import useDocumentationUrl from '@/hooks/use-documentation-url';
import useUserPreferences from '@/hooks/use-user-preferences';

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const BarGraph = (props: SVGProps<SVGSVGElement>) => (
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" {...props}>

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Bolt = (props: SVGProps<SVGSVGElement>) => (
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" {...props}>

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Box = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Cloud = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Connection = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Contact = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Document = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Gear = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const List = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const UserProfile = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,4 @@
import { SVGProps } from 'react';
import type { SVGProps } from 'react';
const Web = (props: SVGProps<SVGSVGElement>) => {
return (

View file

@ -1,4 +1,5 @@
import { useLogto, IdTokenClaims } from '@logto/react';
import type { IdTokenClaims } from '@logto/react';
import { useLogto } from '@logto/react';
import classNames from 'classnames';
import { useEffect, useRef, useState } from 'react';
import { useTranslation } from 'react-i18next';

View file

@ -1,4 +1,5 @@
import { AppearanceMode, ApplicationType } from '@logto/schemas';
import type { ApplicationType } from '@logto/schemas';
import { AppearanceMode } from '@logto/schemas';
import { darkModeApplicationIconMap, lightModeApplicationIconMap } from '@/consts';
import { useTheme } from '@/hooks/use-theme';

View file

@ -1,4 +1,4 @@
import { Application } from '@logto/schemas';
import type { Application } from '@logto/schemas';
import { adminConsoleApplicationId } from '@logto/schemas/lib/seeds';
import { useTranslation } from 'react-i18next';
import { Link } from 'react-router-dom';

View file

@ -1,4 +1,4 @@
import { Application } from '@logto/schemas';
import type { Application } from '@logto/schemas';
import { adminConsoleApplicationId } from '@logto/schemas/lib/seeds';
import { useTranslation } from 'react-i18next';
import useSWR from 'swr';

View file

@ -1,4 +1,5 @@
import { LogDto, LogResult } from '@logto/schemas';
import type { LogDto } from '@logto/schemas';
import { LogResult } from '@logto/schemas';
import { conditionalString } from '@silverhand/essentials';
import classNames from 'classnames';
import { useTranslation } from 'react-i18next';
@ -11,7 +12,7 @@ import TableEmpty from '@/components/Table/TableEmpty';
import TableError from '@/components/Table/TableError';
import TableLoading from '@/components/Table/TableLoading';
import UserName from '@/components/UserName';
import { RequestError } from '@/hooks/use-api';
import type { RequestError } from '@/hooks/use-api';
import * as tableStyles from '@/scss/table.module.scss';
import ApplicationSelector from './components/ApplicationSelector';

View file

@ -1,11 +1,12 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { HTMLProps, ReactElement, ReactNode, useEffect, useRef, useState } from 'react';
import type { HTMLProps, ReactElement, ReactNode } from 'react';
import { useEffect, useRef, useState } from 'react';
import { useTranslation } from 'react-i18next';
import { Ring as Spinner } from '@/components/Spinner';
import DangerousRaw from '../DangerousRaw';
import type DangerousRaw from '../DangerousRaw';
import * as styles from './index.module.scss';
export type ButtonType = 'primary' | 'danger' | 'outline' | 'text' | 'default' | 'branding';

View file

@ -1,5 +1,6 @@
import classNames from 'classnames';
import { forwardRef, Ref, ReactNode } from 'react';
import type { Ref, ReactNode } from 'react';
import { forwardRef } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,9 +1,9 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { ReactElement } from 'react';
import type { ReactElement } from 'react';
import { useTranslation } from 'react-i18next';
import DangerousRaw from '../DangerousRaw';
import type DangerousRaw from '../DangerousRaw';
import * as styles from './index.module.scss';
type Props = {

View file

@ -1,5 +1,6 @@
import { nanoid } from 'nanoid';
import { ReactNode, useState } from 'react';
import type { ReactNode } from 'react';
import { useState } from 'react';
import Icon from './Icon';
import * as styles from './index.module.scss';

View file

@ -1,5 +1,6 @@
import classNames from 'classnames';
import { ChangeEvent, KeyboardEvent, useRef } from 'react';
import type { ChangeEvent, KeyboardEvent } from 'react';
import { useRef } from 'react';
import { PrismAsyncLight as SyntaxHighlighter } from 'react-syntax-highlighter';
import { a11yDark as theme } from 'react-syntax-highlighter/dist/esm/styles/prism';

View file

@ -1,5 +1,6 @@
import { nanoid } from 'nanoid';
import { ChangeEventHandler, useState } from 'react';
import type { ChangeEventHandler } from 'react';
import { useState } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,9 +1,10 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
import ReactModal from 'react-modal';
import Button, { ButtonType } from '@/components/Button';
import type { ButtonType } from '@/components/Button';
import Button from '@/components/Button';
import * as modalStyles from '@/scss/modal.module.scss';
import ModalLayout from '../ModalLayout';

View file

@ -1,6 +1,8 @@
import classNames from 'classnames';
import { MouseEventHandler, useEffect, useMemo, useRef, useState } from 'react';
import { TFuncKey, useTranslation } from 'react-i18next';
import type { MouseEventHandler } from 'react';
import { useEffect, useMemo, useRef, useState } from 'react';
import type { TFuncKey } from 'react-i18next';
import { useTranslation } from 'react-i18next';
import Copy from '@/assets/images/copy.svg';
import EyeClosed from '@/assets/images/eye-closed.svg';

View file

@ -1,4 +1,4 @@
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
type Props = {
children: ReactNode;

View file

@ -1,4 +1,4 @@
import { Nullable } from '@silverhand/essentials';
import type { Nullable } from '@silverhand/essentials';
import dayjs from 'dayjs';
type Props = {

View file

@ -1,4 +1,5 @@
import { ReactNode, useState } from 'react';
import type { ReactNode } from 'react';
import { useState } from 'react';
import ConfirmModal from '@/components/ConfirmModal';
import TextInput from '@/components/TextInput';

View file

@ -1,4 +1,4 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import ReactModal from 'react-modal';
import Close from '@/assets/images/close.svg';

View file

@ -1,5 +1,5 @@
import classNames from 'classnames';
import { MouseEvent, KeyboardEvent, ReactNode } from 'react';
import type { MouseEvent, KeyboardEvent, ReactNode } from 'react';
import { onKeyDownHandler } from '@/utilities/a11y';

View file

@ -1,8 +1,10 @@
import classNames from 'classnames';
import { ReactNode, RefObject, useRef } from 'react';
import type { ReactNode, RefObject } from 'react';
import { useRef } from 'react';
import ReactModal from 'react-modal';
import usePosition, { HorizontalAlignment } from '@/hooks/use-position';
import type { HorizontalAlignment } from '@/hooks/use-position';
import usePosition from '@/hooks/use-position';
import { onKeyDownHandler } from '@/utilities/a11y';
import * as styles from './index.module.scss';

View file

@ -1,5 +1,6 @@
import { conditional } from '@silverhand/essentials';
import { Component, ReactNode } from 'react';
import type { ReactNode } from 'react';
import { Component } from 'react';
import AppError from '../AppError';

View file

@ -1,11 +1,12 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { ReactElement, ReactNode, useRef } from 'react';
import type { ReactElement, ReactNode } from 'react';
import { useRef } from 'react';
import { useTranslation } from 'react-i18next';
import Tip from '@/assets/images/tip.svg';
import DangerousRaw from '../DangerousRaw';
import type DangerousRaw from '../DangerousRaw';
import Spacer from '../Spacer';
import Tooltip from '../Tooltip';
import * as styles from './index.module.scss';

View file

@ -1,7 +1,8 @@
import { AdminConsoleKey } from '@logto/phrases';
import { Nullable } from '@silverhand/essentials';
import type { AdminConsoleKey } from '@logto/phrases';
import type { Nullable } from '@silverhand/essentials';
import classNames from 'classnames';
import { ForwardedRef, forwardRef, HTMLProps, useImperativeHandle, useRef } from 'react';
import type { ForwardedRef, HTMLProps } from 'react';
import { forwardRef, useImperativeHandle, useRef } from 'react';
import { useTranslation } from 'react-i18next';
import Tooltip from '../Tooltip';

View file

@ -1,6 +1,7 @@
import classNames from 'classnames';
import { ReactNode } from 'react';
import { Link, To } from 'react-router-dom';
import type { ReactNode } from 'react';
import type { To } from 'react-router-dom';
import { Link } from 'react-router-dom';
import * as styles from './index.module.scss';

View file

@ -1,10 +1,10 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { ReactElement, ReactNode } from 'react';
import type { ReactElement, ReactNode } from 'react';
import { useTranslation } from 'react-i18next';
import { Link } from 'react-router-dom';
import DangerousRaw from '../DangerousRaw';
import type DangerousRaw from '../DangerousRaw';
import * as styles from './index.module.scss';
type Props = {

View file

@ -1,4 +1,5 @@
import { HTMLProps, useRef, useState } from 'react';
import type { HTMLProps } from 'react';
import { useRef, useState } from 'react';
const githubRawUrlPrefix = 'https://raw.githubusercontent.com/logto-io/logto/master';

View file

@ -1,4 +1,5 @@
import { conditionalString, Optional } from '@silverhand/essentials';
import type { Optional } from '@silverhand/essentials';
import { conditionalString } from '@silverhand/essentials';
import classNames from 'classnames';
import { memo, useRef } from 'react';
import ReactMarkdown from 'react-markdown';

View file

@ -1,6 +1,6 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
import Close from '@/assets/images/close.svg';

View file

@ -1,5 +1,6 @@
import { AdminConsoleKey } from '@logto/phrases';
import { KeyboardEvent, useMemo, useState } from 'react';
import type { AdminConsoleKey } from '@logto/phrases';
import type { KeyboardEvent } from 'react';
import { useMemo, useState } from 'react';
import { useTranslation } from 'react-i18next';
import Minus from '@/assets/images/minus.svg';
@ -9,7 +10,7 @@ import ConfirmModal from '../ConfirmModal';
import IconButton from '../IconButton';
import TextInput from '../TextInput';
import * as styles from './index.module.scss';
import { MultiTextInputError } from './types';
import type { MultiTextInputError } from './types';
type Props = {
title: AdminConsoleKey;
@ -53,7 +54,7 @@ const MultiTextInput = ({ title, value, onChange, onKeyPress, error, placeholder
<div className={styles.deletableInput}>
<TextInput
hasError={Boolean(
error?.inputs?.[fieldIndex] || (fieldIndex === 0 && error?.required)
error?.inputs?.[fieldIndex] ?? (fieldIndex === 0 && error?.required)
)}
value={fieldValue}
placeholder={placeholder}

View file

@ -2,7 +2,8 @@ import { t } from 'i18next';
import { safeParseJson } from '@/utilities/json';
import { MultiTextInputError, multiTextInputErrorGuard, MultiTextInputRule } from './types';
import type { MultiTextInputError, MultiTextInputRule } from './types';
import { multiTextInputErrorGuard } from './types';
export const validate = (
value?: string[],

View file

@ -1,6 +1,7 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import { KeyboardEventHandler, ReactNode, useCallback } from 'react';
import type { KeyboardEventHandler, ReactNode } from 'react';
import { useCallback } from 'react';
import { useTranslation } from 'react-i18next';
import * as styles from './index.module.scss';

View file

@ -1,7 +1,9 @@
import classNames from 'classnames';
import { Children, cloneElement, forwardRef, isValidElement, LegacyRef, ReactNode } from 'react';
import type { LegacyRef, ReactNode } from 'react';
import { Children, cloneElement, forwardRef, isValidElement } from 'react';
import Radio, { Props as RadioProps } from './Radio';
import type { Props as RadioProps } from './Radio';
import Radio from './Radio';
import * as styles from './index.module.scss';
type Props = {

View file

@ -1,4 +1,5 @@
import { FormEventHandler, KeyboardEventHandler, useState } from 'react';
import type { FormEventHandler, KeyboardEventHandler } from 'react';
import { useState } from 'react';
import { useTranslation } from 'react-i18next';
import SearchIcon from '@/assets/images/search.svg';

View file

@ -1,5 +1,6 @@
import classNames from 'classnames';
import { ReactEventHandler, ReactNode, useRef, useState } from 'react';
import type { ReactEventHandler, ReactNode } from 'react';
import { useRef, useState } from 'react';
import Close from '@/assets/images/close.svg';
import KeyboardArrowDown from '@/assets/images/keyboard-arrow-down.svg';

View file

@ -1,5 +1,5 @@
import classNames from 'classnames';
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,4 +1,5 @@
import { forwardRef, HTMLProps, ReactNode, Ref } from 'react';
import type { HTMLProps, ReactNode, Ref } from 'react';
import { forwardRef } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,5 +1,5 @@
import { AppearanceMode } from '@logto/schemas';
import { ReactNode } from 'react';
import type { ReactNode } from 'react';
import { useTranslation } from 'react-i18next';
import EmptyDark from '@/assets/images/table-empty-dark.svg';

View file

@ -1,5 +1,6 @@
import classNames from 'classnames';
import { forwardRef, HTMLProps, ReactNode, ForwardedRef } from 'react';
import type { HTMLProps, ReactNode, ForwardedRef } from 'react';
import { forwardRef } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,5 +1,6 @@
import classNames from 'classnames';
import { ForwardedRef, forwardRef, HTMLProps } from 'react';
import type { ForwardedRef, HTMLProps } from 'react';
import { forwardRef } from 'react';
import * as styles from './index.module.scss';

View file

@ -1,8 +1,10 @@
import classNames from 'classnames';
import { ReactNode, RefObject, useEffect, useLayoutEffect, useRef, useState } from 'react';
import type { ReactNode, RefObject } from 'react';
import { useEffect, useLayoutEffect, useRef, useState } from 'react';
import { createPortal } from 'react-dom';
import usePosition, { VerticalAlignment, HorizontalAlignment } from '@/hooks/use-position';
import type { VerticalAlignment, HorizontalAlignment } from '@/hooks/use-position';
import usePosition from '@/hooks/use-position';
import * as styles from './index.module.scss';

View file

@ -1,5 +1,6 @@
import { conditionalString } from '@silverhand/essentials';
import { createContext, ReactNode, useMemo, useState } from 'react';
import type { ReactNode } from 'react';
import { createContext, useMemo, useState } from 'react';
import { DndProvider } from 'react-dnd';
import { HTML5Backend } from 'react-dnd-html5-backend';

View file

@ -1,6 +1,7 @@
import { Nullable } from '@silverhand/essentials';
import type { Nullable } from '@silverhand/essentials';
import type { Identifier } from 'dnd-core';
import { ReactNode, useContext, useEffect, useRef } from 'react';
import type { ReactNode } from 'react';
import { useContext, useEffect, useRef } from 'react';
import { useDrag, useDrop } from 'react-dnd';
import { DragDropContext } from './DragDropProvider';

View file

@ -1,5 +1,6 @@
import classNames from 'classnames';
import { ReactNode, useMemo } from 'react';
import type { ReactNode } from 'react';
import { useMemo } from 'react';
import { useTranslation } from 'react-i18next';
import CircleMinus from '@/assets/images/circle-minus.svg';

View file

@ -1,7 +1,8 @@
import type { Blocker, Transition } from 'history';
import { useCallback, useContext, useLayoutEffect, useState } from 'react';
import { useTranslation } from 'react-i18next';
import { UNSAFE_NavigationContext, Navigator } from 'react-router-dom';
import type { Navigator } from 'react-router-dom';
import { UNSAFE_NavigationContext } from 'react-router-dom';
import ConfirmModal from '../ConfirmModal';

View file

@ -1,9 +1,10 @@
import { User, UserRole } from '@logto/schemas';
import type { User } from '@logto/schemas';
import { UserRole } from '@logto/schemas';
import { useTranslation } from 'react-i18next';
import { Link } from 'react-router-dom';
import useSWR from 'swr';
import { RequestError } from '@/hooks/use-api';
import type { RequestError } from '@/hooks/use-api';
import * as styles from './index.module.scss';

View file

@ -1,4 +1,4 @@
import { AdminConsoleKey } from '@logto/phrases';
import type { AdminConsoleKey } from '@logto/phrases';
import { ConnectorPlatform, ConnectorType } from '@logto/schemas';
import EmailConnector from '@/assets/images/connector-email.svg';

View file

@ -1,5 +1,5 @@
import { useLogto } from '@logto/react';
import { RequestErrorBody } from '@logto/schemas';
import type { RequestErrorBody } from '@logto/schemas';
import { managementResource } from '@logto/schemas/lib/seeds';
import ky from 'ky';
import { useMemo } from 'react';

View file

@ -1,9 +1,10 @@
import { ConnectorResponse, ConnectorType } from '@logto/schemas';
import type { ConnectorResponse } from '@logto/schemas';
import { ConnectorType } from '@logto/schemas';
import { useMemo } from 'react';
import useSWR from 'swr';
import { RequestError } from '@/hooks/use-api';
import { ConnectorGroup } from '@/types/connector';
import type { RequestError } from '@/hooks/use-api';
import type { ConnectorGroup } from '@/types/connector';
// Group connectors by target
const useConnectorGroups = () => {

View file

@ -1,7 +1,8 @@
import { ConnectorType, SignInExperience, SignInMethodState } from '@logto/schemas';
import type { SignInExperience } from '@logto/schemas';
import { ConnectorType, SignInMethodState } from '@logto/schemas';
import useSWR from 'swr';
import { RequestError } from './use-api';
import type { RequestError } from './use-api';
const useConnectorInUse = (type?: ConnectorType, target?: string): boolean | undefined => {
const { data } = useSWR<SignInExperience, RequestError>(target && type && '/api/sign-in-exp');

View file

@ -1,4 +1,5 @@
import { RefObject, useCallback, useEffect, useState } from 'react';
import type { RefObject } from 'react';
import { useCallback, useEffect, useState } from 'react';
export type VerticalAlignment = 'top' | 'center' | 'bottom';

View file

@ -1,4 +1,4 @@
import { Nullable } from '@silverhand/essentials';
import type { Nullable } from '@silverhand/essentials';
import { useState, useEffect, useCallback } from 'react';
const useScroll = (contentRef: Nullable<HTMLDivElement>) => {

View file

@ -1,8 +1,9 @@
import { useLogto } from '@logto/react';
import { AdminConsoleConfig, Setting } from '@logto/schemas';
import type { AdminConsoleConfig, Setting } from '@logto/schemas';
import useSWR from 'swr';
import useApi, { RequestError } from './use-api';
import type { RequestError } from './use-api';
import useApi from './use-api';
const useSettings = () => {
const { isAuthenticated, error: authError } = useLogto();

View file

@ -1,8 +1,8 @@
import { RequestErrorBody } from '@logto/schemas';
import type { RequestErrorBody } from '@logto/schemas';
import { HTTPError } from 'ky';
import { useCallback } from 'react';
import { useTranslation } from 'react-i18next';
import { BareFetcher } from 'swr';
import type { BareFetcher } from 'swr';
import useApi, { RequestError } from './use-api';

View file

@ -1,4 +1,4 @@
import { SWRConfiguration } from 'swr';
import type { SWRConfiguration } from 'swr';
import { RequestError } from './use-api';
import useSwrFetcher from './use-swr-fetcher';

View file

@ -1,11 +1,12 @@
import { LanguageTag } from '@logto/language-kit';
import type { LanguageTag } from '@logto/language-kit';
import { builtInLanguages as builtInUiLanguages } from '@logto/phrases-ui';
import { useCallback, useMemo } from 'react';
import useSWR from 'swr';
import { CustomPhraseResponse } from '@/types/custom-phrase';
import type { CustomPhraseResponse } from '@/types/custom-phrase';
import useApi, { RequestError } from './use-api';
import type { RequestError } from './use-api';
import useApi from './use-api';
const useUiLanguages = () => {
const {

View file

@ -1,7 +1,7 @@
import { builtInLanguages as builtInConsoleLanguages } from '@logto/phrases';
import { useLogto } from '@logto/react';
import { AppearanceMode } from '@logto/schemas';
import { Nullable, Optional } from '@silverhand/essentials';
import type { Nullable, Optional } from '@silverhand/essentials';
import { t } from 'i18next';
import { useCallback, useEffect, useMemo } from 'react';
import { toast } from 'react-hot-toast';
@ -10,7 +10,8 @@ import { z } from 'zod';
import { themeStorageKey } from '@/consts';
import useApi, { RequestError } from './use-api';
import type { RequestError } from './use-api';
import useApi from './use-api';
import useLogtoUserId from './use-logto-user-id';
const userPreferencesGuard = z.object({

View file

@ -1,4 +1,4 @@
import { LanguageTag } from '@logto/language-kit';
import type { LanguageTag } from '@logto/language-kit';
import resources from '@logto/phrases';
import i18next from 'i18next';
import LanguageDetector from 'i18next-browser-languagedetector';

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