diff --git a/.env.template b/.env.template
deleted file mode 100644
index 4c8ede67baf33..0000000000000
--- a/.env.template
+++ /dev/null
@@ -1,7 +0,0 @@
-CHANGELOG_URL=
-ENABLE_NEW_SETTING_UNSTABLE_API=
-ENABLE_CAPTCHA=
-CAPTCHA_SITE_KEY=
-ENABLE_ENHANCE_SHARE_MODE=
-ALLOW_LOCAL_WORKSPACE=
-DEBUG_JOTAI=
\ No newline at end of file
diff --git a/.github/workflows/build-test.yml b/.github/workflows/build-test.yml
index c6d34e549e316..6a22f9bf073aa 100644
--- a/.github/workflows/build-test.yml
+++ b/.github/workflows/build-test.yml
@@ -280,8 +280,7 @@ jobs:
electron-install: false
full-cache: true
- name: Build Electron renderer
- # always skip cache because its fast, and cache configuration is always changing
- run: yarn build
+ run: yarn affine @affine/electron bundle
env:
DISTRIBUTION: desktop
- name: zip web
diff --git a/.gitignore b/.gitignore
index a2a3f8492a044..cf88160ead31d 100644
--- a/.gitignore
+++ b/.gitignore
@@ -81,3 +81,9 @@ apps/web/next-routes.conf
packages/frontend/templates/edgeless
packages/frontend/core/public/static/templates
+
+# new dist folder
+tsbuild/
+# script
+af
+affine
diff --git a/.prettierignore b/.prettierignore
index 6aa0948e1fdde..59b17bb746f0c 100644
--- a/.prettierignore
+++ b/.prettierignore
@@ -25,4 +25,5 @@ packages/frontend/templates/onboarding
packages/backend/native/index.d.ts
packages/frontend/native/index.d.ts
packages/frontend/native/index.js
-compose.yaml
\ No newline at end of file
+compose.yaml
+*.gen.ts
\ No newline at end of file
diff --git a/eslint.config.mjs b/eslint.config.mjs
index 04b636602900c..764923081b672 100644
--- a/eslint.config.mjs
+++ b/eslint.config.mjs
@@ -31,6 +31,7 @@ export default tseslint.config(
'.yarn/**',
'**/*.d.ts',
'.github/**/*',
+ '**/*.gen.ts',
'packages/frontend/component/.storybook/**/*',
'packages/frontend/i18n/src/i18n-generated.ts',
'packages/frontend/i18n/src/i18n-completenesses.json',
@@ -216,7 +217,7 @@ export default tseslint.config(
},
},
{
- files: ['packages/**/*.{ts,tsx}', 'tools/cli/**/*.{ts,tsx}'],
+ files: ['packages/**/*.{ts,tsx}', 'tools/**/*.ts'],
rules: {
'@typescript-eslint/no-floating-promises': [
'error',
diff --git a/oxlint.json b/oxlint.json
index 45aa4926f8627..12cf04135cf14 100644
--- a/oxlint.json
+++ b/oxlint.json
@@ -5,7 +5,7 @@
"correctness": "error",
"perf": "error"
},
- "ignorePatterns": ["tools/cli/src/webpack/error-handler.js"],
+ "ignorePatterns": ["tools/scripts/src/webpack/error-handler.js"],
"rules": {
"import/named": "allow",
"no-await-in-loop": "allow",
@@ -167,7 +167,8 @@
"files": [
"*.{spec,test,e2e,stories}.{ts,tsx}",
"tests/**/*.ts",
- "packages/backend/server/tests/**/*.ts"
+ "packages/backend/server/tests/**/*.ts",
+ "tools/**.*"
],
"rules": {
"typescript/no-non-null-assertion": "off",
diff --git a/package.json b/package.json
index 57bb522fd6903..17ab671b9e8dc 100644
--- a/package.json
+++ b/package.json
@@ -17,12 +17,10 @@
"node": "<21.0.0"
},
"scripts": {
- "dev": "yarn workspace @affine/cli dev",
- "build": "yarn workspace @affine/cli bundle",
- "dev:electron": "yarn workspace @affine/electron dev",
- "build:electron": "yarn workspace @affine/electron build",
- "build:server-native": "yarn workspace @affine/server-native build",
- "start:web-static": "yarn workspace @affine/web static-server",
+ "affine": "yarn workspace @affine-tools/scripts affine",
+ "af": "yarn workspace @affine-tools/scripts af",
+ "dev": "yarn affine dev",
+ "build": "yarn affine build",
"lint:eslint": "cross-env NODE_OPTIONS=\"--max-old-space-size=8192\" eslint --report-unused-disable-directives-severity=off . --cache",
"lint:eslint:fix": "yarn lint:eslint --fix",
"lint:prettier": "prettier --ignore-unknown --cache --check .",
@@ -33,9 +31,8 @@
"test": "vitest --run",
"test:ui": "vitest --ui",
"test:coverage": "vitest run --coverage",
- "typecheck": "tsc -b tsconfig.json",
- "postinstall": "node ./scripts/check-version.mjs && yarn workspace @affine/i18n i18n-codegen gen && yarn husky install",
- "prepare": "husky"
+ "typecheck": "tsc -b tsconfig.project.json --verbose",
+ "postinstall": "yarn affine codegen && yarn husky"
},
"lint-staged": {
"*": "prettier --write --ignore-unknown --cache",
@@ -51,7 +48,7 @@
]
},
"devDependencies": {
- "@affine/cli": "workspace:*",
+ "@affine-tools/scripts": "workspace:*",
"@capacitor/cli": "^6.2.0",
"@eslint/js": "^9.16.0",
"@faker-js/faker": "^9.3.0",
diff --git a/packages/frontend/admin/package.json b/packages/frontend/admin/package.json
index 9f969a735ef71..fa8bd3a5b810a 100644
--- a/packages/frontend/admin/package.json
+++ b/packages/frontend/admin/package.json
@@ -62,7 +62,8 @@
"tailwindcss-animate": "^1.0.7"
},
"scripts": {
- "build": "cross-env DISTRIBUTION=admin yarn workspace @affine/cli bundle",
+ "build": "affine bundle",
+ "dev": "affine bundle --dev",
"update-shadcn": "shadcn-ui add -p src/components/ui"
},
"exports": {
diff --git a/packages/frontend/admin/tailwind.config.js b/packages/frontend/admin/tailwind.config.js
index ee3b0653cb636..f433890cff0f8 100644
--- a/packages/frontend/admin/tailwind.config.js
+++ b/packages/frontend/admin/tailwind.config.js
@@ -1,7 +1,8 @@
/** @type {import('tailwindcss').Config} */
module.exports = {
darkMode: ['class'],
- content: ['./src/**/*.{ts,tsx}'],
+ // TODO(@forehalo): we are not running webpack in admin dir
+ content: ['./packages/frontend/admin/src/**/*.{ts,tsx}'],
prefix: '',
theme: {
container: {
diff --git a/packages/frontend/apps/android/package.json b/packages/frontend/apps/android/package.json
index d8e8832fd6635..97220a334ab7b 100644
--- a/packages/frontend/apps/android/package.json
+++ b/packages/frontend/apps/android/package.json
@@ -5,9 +5,8 @@
"private": true,
"browser": "src/index.tsx",
"scripts": {
- "build": "cross-env DISTRIBUTION=android yarn workspace @affine/cli bundle",
- "dev": "yarn workspace @affine/cli dev",
- "static-server": "cross-env DISTRIBUTION=android yarn workspace @affine/cli dev --static"
+ "build": "affine bundle",
+ "dev": "affine bundle --dev"
},
"dependencies": {
"@affine/component": "workspace:*",
diff --git a/packages/frontend/apps/electron/package.json b/packages/frontend/apps/electron/package.json
index a20be5b11a5d8..acef6bbf29a0c 100644
--- a/packages/frontend/apps/electron/package.json
+++ b/packages/frontend/apps/electron/package.json
@@ -2,28 +2,23 @@
"name": "@affine/electron",
"private": true,
"version": "0.18.0",
- "author": "toeverything",
- "repository": {
- "url": "https://github.com/toeverything/AFFiNE",
- "type": "git"
- },
- "description": "AFFiNE Desktop App",
- "homepage": "https://github.com/toeverything/AFFiNE",
+ "main": "./dist/main.js",
"scripts": {
"start": "electron .",
- "dev": "DEV_SERVER_URL=http://localhost:8080 node --loader ts-node/esm/transpile-only ./scripts/dev.ts",
- "dev:prod": "yarn node --loader ts-node/esm/transpile-only scripts/dev.ts",
- "build": "NODE_ENV=production node --loader ts-node/esm/transpile-only scripts/build-layers.ts",
- "build:dev": "NODE_ENV=development node --loader ts-node/esm/transpile-only scripts/build-layers.ts",
- "generate-assets": "node --loader ts-node/esm/transpile-only scripts/generate-assets.ts",
- "package": "cross-env NODE_OPTIONS=\"--loader ts-node/esm/transpile-only\" electron-forge package",
- "make": "cross-env NODE_OPTIONS=\"--loader ts-node/esm/transpile-only\" electron-forge make",
- "make-squirrel": "node --loader ts-node/esm/transpile-only scripts/make-squirrel.ts",
- "make-nsis": "node --loader ts-node/esm/transpile-only scripts/make-nsis.ts"
+ "dev": "cross-env DEV_SERVER_URL=http://localhost:8080 node ./scripts/dev.ts",
+ "dev:prod": "node ./scripts/dev.ts",
+ "build": "cross-env NODE_ENV=production node ./scripts/build-layers.ts",
+ "build:dev": "node ./scripts/build-layers.ts",
+ "bundle": "affine bundle",
+ "generate-assets": "node ./scripts/generate-assets.ts",
+ "package": "electron-forge package",
+ "make": "electron-forge make",
+ "make-squirrel": "node ./scripts/make-squirrel.ts",
+ "make-nsis": "node ./scripts/make-nsis.ts"
},
- "main": "./dist/main.js",
"devDependencies": {
"@affine-test/kit": "workspace:*",
+ "@affine-tools/utils": "workspace:*",
"@affine/component": "workspace:*",
"@affine/core": "workspace:*",
"@affine/electron-api": "workspace:*",
diff --git a/packages/frontend/apps/electron/scripts/common.ts b/packages/frontend/apps/electron/scripts/common.ts
index ddc10995088f0..5775953a16c13 100644
--- a/packages/frontend/apps/electron/scripts/common.ts
+++ b/packages/frontend/apps/electron/scripts/common.ts
@@ -1,8 +1,8 @@
import { resolve } from 'node:path';
import { fileURLToPath } from 'node:url';
-// eslint-disable-next-line @typescript-eslint/no-restricted-imports
-import { getBuildConfig } from '@affine/cli/src/webpack/runtime-config';
+import { getBuildConfig } from '@affine-tools/utils/build-config';
+import { Workspace } from '@affine-tools/utils/workspace';
import { sentryEsbuildPlugin } from '@sentry/esbuild-plugin';
import type { BuildOptions, Plugin } from 'esbuild';
@@ -24,12 +24,10 @@ export const config = (): BuildOptions => {
'process.env.NODE_ENV': process.env.NODE_ENV,
REPLACE_ME_BUILD_ENV: process.env.BUILD_TYPE ?? 'stable',
...Object.entries(
- getBuildConfig({
- channel: (process.env.BUILD_TYPE as any) ?? 'canary',
- distribution: 'desktop',
+ getBuildConfig(new Workspace().getPackage('@affine/electron'), {
mode:
process.env.NODE_ENV === 'production' ? 'production' : 'development',
- static: false,
+ channel: (process.env.BUILD_TYPE as any) ?? 'canary',
})
).reduce(
(def, [key, val]) => {
diff --git a/packages/frontend/apps/electron/tsconfig.json b/packages/frontend/apps/electron/tsconfig.json
index df1043bd00f99..2b455fb9eaf05 100644
--- a/packages/frontend/apps/electron/tsconfig.json
+++ b/packages/frontend/apps/electron/tsconfig.json
@@ -33,6 +33,9 @@
},
{
"path": "../../../../tests/kit"
+ },
+ {
+ "path": "../../../../tools/utils"
}
],
"ts-node": {
diff --git a/packages/frontend/apps/electron/tsconfig.node.json b/packages/frontend/apps/electron/tsconfig.node.json
index 2beac5166efc1..2fcb612fd0800 100644
--- a/packages/frontend/apps/electron/tsconfig.node.json
+++ b/packages/frontend/apps/electron/tsconfig.node.json
@@ -5,7 +5,7 @@
"target": "ESNext",
"module": "ESNext",
"resolveJsonModule": true,
- "moduleResolution": "Node",
+ "moduleResolution": "bundler",
"allowSyntheticDefaultImports": true,
"noEmit": false,
"outDir": "./lib/scripts",
@@ -17,5 +17,10 @@
"ts-node": {
"esm": true,
"experimentalSpecifierResolution": "node"
- }
+ },
+ "references": [
+ {
+ "path": "../../../../tools/utils"
+ }
+ ]
}
diff --git a/packages/frontend/apps/electron/webpack.config.ts b/packages/frontend/apps/electron/webpack.config.ts
new file mode 100644
index 0000000000000..6c2c6e2fa1b06
--- /dev/null
+++ b/packages/frontend/apps/electron/webpack.config.ts
@@ -0,0 +1,7 @@
+// TODO(@forehalo): all packages would become 'module'
+module.exports.config = {
+ entry: {
+ app: './renderer/index.tsx',
+ shell: './renderer/shell/index.tsx',
+ },
+};
diff --git a/packages/frontend/apps/ios/package.json b/packages/frontend/apps/ios/package.json
index c3aa1b6c1f38a..78a99fec7c5e5 100644
--- a/packages/frontend/apps/ios/package.json
+++ b/packages/frontend/apps/ios/package.json
@@ -5,11 +5,10 @@
"private": true,
"browser": "src/index.tsx",
"scripts": {
- "build": "cross-env DISTRIBUTION=ios yarn workspace @affine/cli bundle",
- "dev": "yarn workspace @affine/cli dev",
+ "build": "affine bundle",
+ "dev": "affine bundle --dev",
"sync": "yarn cap sync",
- "sync:dev": "CAP_SERVER_URL=http://localhost:8080 yarn cap sync",
- "static-server": "cross-env DISTRIBUTION=ios yarn workspace @affine/cli dev --static"
+ "sync:dev": "CAP_SERVER_URL=http://localhost:8080 yarn cap sync"
},
"dependencies": {
"@affine/component": "workspace:*",
diff --git a/packages/frontend/apps/mobile/package.json b/packages/frontend/apps/mobile/package.json
index ae1d786897b6a..f2d859da27ff3 100644
--- a/packages/frontend/apps/mobile/package.json
+++ b/packages/frontend/apps/mobile/package.json
@@ -5,9 +5,8 @@
"private": true,
"browser": "src/index.tsx",
"scripts": {
- "build": "cross-env DISTRIBUTION=mobile yarn workspace @affine/cli bundle",
- "dev": "yarn workspace @affine/cli dev",
- "static-server": "cross-env DISTRIBUTION=mobile yarn workspace @affine/cli dev --static"
+ "build": "affine bundle",
+ "dev": "affine bundle --dev"
},
"dependencies": {
"@affine/component": "workspace:*",
diff --git a/packages/frontend/apps/web/package.json b/packages/frontend/apps/web/package.json
index b3a995920f089..d60092b257f6e 100644
--- a/packages/frontend/apps/web/package.json
+++ b/packages/frontend/apps/web/package.json
@@ -5,9 +5,8 @@
"private": true,
"browser": "src/index.tsx",
"scripts": {
- "build": "cross-env DISTRIBUTION=web yarn workspace @affine/cli bundle",
- "dev": "yarn workspace @affine/cli dev",
- "static-server": "yarn workspace @affine/cli dev --static"
+ "build": "affine bundle",
+ "dev": "affine bundle --dev"
},
"dependencies": {
"@affine/component": "workspace:*",
diff --git a/packages/frontend/component/.storybook/main.ts b/packages/frontend/component/.storybook/main.ts
index ac46638d1d16a..9da8b74fc958e 100644
--- a/packages/frontend/component/.storybook/main.ts
+++ b/packages/frontend/component/.storybook/main.ts
@@ -3,7 +3,7 @@ import { StorybookConfig } from '@storybook/react-vite';
import { vanillaExtractPlugin } from '@vanilla-extract/vite-plugin';
import swc from 'unplugin-swc';
import { mergeConfig } from 'vite';
-import { getBuildConfig } from '@affine/cli/src/webpack/runtime-config';
+import { getBuildConfig } from '@affine-tools/scripts/src/webpack/runtime-config';
export default {
stories: ['../src/ui/**/*.@(mdx|stories.@(js|jsx|ts|tsx))'],
diff --git a/packages/frontend/component/package.json b/packages/frontend/component/package.json
index bb20658c7bcee..9577f34adee44 100644
--- a/packages/frontend/component/package.json
+++ b/packages/frontend/component/package.json
@@ -20,7 +20,7 @@
"react-dom": "^19.0.0"
},
"dependencies": {
- "@affine/cli": "workspace:*",
+ "@affine-tools/scripts": "workspace:*",
"@affine/debug": "workspace:*",
"@affine/electron-api": "workspace:*",
"@affine/graphql": "workspace:*",
diff --git a/packages/frontend/core/public/robots.txt b/packages/frontend/core/public/robots.txt
index 7b537c62f904c..66864824ccdb4 100644
--- a/packages/frontend/core/public/robots.txt
+++ b/packages/frontend/core/public/robots.txt
@@ -2,6 +2,7 @@
User-agent: *
Allow: /
+Allow: /api/workspaces/*/blobs/*
Disallow: /oauth/*
Disallow: /workspace/*
Disallow: /public-workspace/*
diff --git a/packages/frontend/core/tsconfig.node.json b/packages/frontend/core/tsconfig.node.json
index acbcfbefb94c4..22e466d88025e 100644
--- a/packages/frontend/core/tsconfig.node.json
+++ b/packages/frontend/core/tsconfig.node.json
@@ -13,9 +13,6 @@
},
"include": [".webpack/*.ts"],
"references": [
- {
- "path": "../../../tools/cli"
- },
{
"path": "../../common/env"
}
diff --git a/packages/frontend/graphql/package.json b/packages/frontend/graphql/package.json
index 1aad8f3c8f8ee..706ceb2418375 100644
--- a/packages/frontend/graphql/package.json
+++ b/packages/frontend/graphql/package.json
@@ -21,7 +21,7 @@
"vitest": "2.1.8"
},
"scripts": {
- "postinstall": "gql-gen --errors-only"
+ "build": "gql-gen --errors-only"
},
"dependencies": {
"@affine/env": "workspace:*",
diff --git a/packages/frontend/templates/package.json b/packages/frontend/templates/package.json
index 6646cb336f90a..698643e1259e2 100644
--- a/packages/frontend/templates/package.json
+++ b/packages/frontend/templates/package.json
@@ -4,7 +4,8 @@
"sideEffect": false,
"version": "0.18.0",
"scripts": {
- "postinstall": "node ./build-edgeless.mjs && node ./build-stickers.mjs"
+ "build": "node ./build-edgeless.mjs && node ./build-stickers.mjs",
+ "postinstall": "yarn build"
},
"type": "module",
"exports": {
diff --git a/scripts/setup/global.ts b/scripts/setup/global.ts
index cedc216dd425e..cc268a8ccb7e8 100644
--- a/scripts/setup/global.ts
+++ b/scripts/setup/global.ts
@@ -1,12 +1,14 @@
-import { getBuildConfig } from '@affine/cli/src/webpack/runtime-config';
import { setupGlobal } from '@affine/env/global';
+import { getBuildConfig } from '@affine-tools/utils/build-config';
+import { Workspace } from '@affine-tools/utils/workspace';
-globalThis.BUILD_CONFIG = getBuildConfig({
- distribution: 'web',
- mode: 'development',
- channel: 'canary',
- static: false,
-});
+globalThis.BUILD_CONFIG = getBuildConfig(
+ new Workspace().getPackage('@affine/web'),
+ {
+ mode: 'development',
+ channel: 'canary',
+ }
+);
if (typeof window !== 'undefined') {
window.location.search = '?prefixUrl=http://127.0.0.1:3010/';
diff --git a/tests/affine-cloud-copilot/playwright.config.ts b/tests/affine-cloud-copilot/playwright.config.ts
index 4b6736354bba5..faff4e14eb90f 100644
--- a/tests/affine-cloud-copilot/playwright.config.ts
+++ b/tests/affine-cloud-copilot/playwright.config.ts
@@ -26,9 +26,8 @@ const config: PlaywrightTestConfig = {
retries: 3,
reporter: process.env.CI ? 'github' : 'list',
webServer: [
- // Intentionally not building the web, reminds you to run it by yourself.
{
- command: 'yarn -T run start:web-static',
+ command: 'yarn run -T affine dev -p @affine/web',
port: 8080,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
@@ -37,7 +36,7 @@ const config: PlaywrightTestConfig = {
},
},
{
- command: 'yarn workspace @affine/server start',
+ command: 'yarn run -T affine dev -p @affine/server',
port: 3010,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
diff --git a/tests/affine-cloud/package.json b/tests/affine-cloud/package.json
index 67995a5496ada..c7d0a614373fe 100644
--- a/tests/affine-cloud/package.json
+++ b/tests/affine-cloud/package.json
@@ -6,6 +6,7 @@
},
"devDependencies": {
"@affine-test/kit": "workspace:*",
+ "@affine-tools/scripts": "workspace:*",
"@playwright/test": "=1.49.1"
},
"version": "0.18.0"
diff --git a/tests/affine-cloud/playwright.config.ts b/tests/affine-cloud/playwright.config.ts
index 68e8d852e5e3f..59d59729b62c6 100644
--- a/tests/affine-cloud/playwright.config.ts
+++ b/tests/affine-cloud/playwright.config.ts
@@ -26,9 +26,12 @@ const config: PlaywrightTestConfig = {
retries: process.env.COPILOT ? 1 : 3,
reporter: process.env.CI ? 'github' : 'list',
webServer: [
- // Intentionally not building the web, reminds you to run it by yourself.
{
- command: 'yarn -T run start:web-static',
+ // TODO(@forehalo):
+ // in ci, all the target will be built,
+ // we could download the builds from archives
+ // and then run the web with simple http serve, it's will be faster
+ command: 'yarn run -T affine dev -p @affine/web',
port: 8080,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
@@ -37,12 +40,10 @@ const config: PlaywrightTestConfig = {
},
},
{
- command: 'yarn workspace @affine/server start',
+ command: 'yarn run -T affine dev -p @affine/server',
port: 3010,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
- stdout: 'pipe',
- stderr: 'pipe',
env: {
DATABASE_URL:
process.env.DATABASE_URL ??
diff --git a/tests/affine-desktop-cloud/playwright.config.ts b/tests/affine-desktop-cloud/playwright.config.ts
index 3432035187049..7765e1bc628b6 100644
--- a/tests/affine-desktop-cloud/playwright.config.ts
+++ b/tests/affine-desktop-cloud/playwright.config.ts
@@ -23,7 +23,7 @@ const config: PlaywrightTestConfig = {
webServer: [
// Intentionally not building the web, reminds you to run it by yourself.
{
- command: 'yarn -T run start:web-static',
+ command: 'yarn run -T affine dev -p @affine/web',
port: 8080,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
@@ -34,7 +34,7 @@ const config: PlaywrightTestConfig = {
},
},
{
- command: 'yarn workspace @affine/server start',
+ command: 'yarn run -T affine dev -p @affine/server',
port: 3010,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
diff --git a/tests/affine-desktop/playwright.config.ts b/tests/affine-desktop/playwright.config.ts
index f6cd39740d55d..ffa4003023c48 100644
--- a/tests/affine-desktop/playwright.config.ts
+++ b/tests/affine-desktop/playwright.config.ts
@@ -41,7 +41,7 @@ if (process.env.DEV_SERVER_URL) {
);
config.webServer = [
{
- command: 'yarn run start:web-static',
+ command: 'yarn run -T affine dev -p @affine/web',
port: 8080,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
diff --git a/tests/affine-local/playwright.config.ts b/tests/affine-local/playwright.config.ts
index 8337cb53b111a..7f68d6a403089 100644
--- a/tests/affine-local/playwright.config.ts
+++ b/tests/affine-local/playwright.config.ts
@@ -46,7 +46,7 @@ const config: PlaywrightTestConfig = {
webServer: [
// Intentionally not building the web, reminds you to run it by yourself.
{
- command: 'yarn run start:web-static',
+ command: 'yarn run -T affine dev -p @affine/web',
port: 8080,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
diff --git a/tests/affine-mobile/playwright.config.ts b/tests/affine-mobile/playwright.config.ts
index 6804256482284..76b381bc588f9 100644
--- a/tests/affine-mobile/playwright.config.ts
+++ b/tests/affine-mobile/playwright.config.ts
@@ -49,7 +49,7 @@ const config: PlaywrightTestConfig = {
webServer: [
// Intentionally not building the web, reminds you to run it by yourself.
{
- command: 'yarn workspace @affine/mobile static-server',
+ command: 'yarn run -T affine dev -p @affine/mobile',
port: 8080,
timeout: 120 * 1000,
reuseExistingServer: !process.env.CI,
diff --git a/tools/cli/src/bin/build.ts b/tools/cli/src/bin/build.ts
deleted file mode 100644
index 64113974a61dd..0000000000000
--- a/tools/cli/src/bin/build.ts
+++ /dev/null
@@ -1,66 +0,0 @@
-import { spawn } from 'node:child_process';
-
-import webpack from 'webpack';
-
-import { getCwdFromDistribution } from '../config/cwd.cjs';
-import type { BuildFlags } from '../config/index.js';
-import { createWebpackConfig } from '../webpack/webpack.config.js';
-
-// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
-const buildType = process.env.BUILD_TYPE_OVERRIDE || process.env.BUILD_TYPE;
-
-if (process.env.BUILD_TYPE_OVERRIDE) {
- process.env.BUILD_TYPE = process.env.BUILD_TYPE_OVERRIDE;
-}
-
-const getChannel = () => {
- switch (buildType) {
- case 'canary':
- case 'beta':
- case 'stable':
- case 'internal':
- return buildType;
- case '':
- throw new Error('BUILD_TYPE is not set');
- default: {
- throw new Error(
- `BUILD_TYPE must be one of canary, beta, stable, internal, received [${buildType}]`
- );
- }
- }
-};
-
-let entry: BuildFlags['entry'];
-
-const { DISTRIBUTION = 'web' } = process.env;
-
-const cwd = getCwdFromDistribution(DISTRIBUTION);
-
-if (DISTRIBUTION === 'desktop') {
- entry = { app: './index.tsx', shell: './shell/index.tsx' };
-}
-
-const flags = {
- distribution: DISTRIBUTION as BuildFlags['distribution'],
- mode: 'production',
- channel: getChannel(),
- coverage: process.env.COVERAGE === 'true',
- entry,
- static: false,
-} satisfies BuildFlags;
-
-spawn('yarn', ['workspace', '@affine/i18n', 'build'], {
- stdio: 'inherit',
-});
-
-// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
-webpack(createWebpackConfig(cwd!, flags), (err, stats) => {
- if (err) {
- console.error(err);
- process.exit(1);
- }
- if (stats?.hasErrors()) {
- console.error(stats.toString('errors-only'));
- process.exit(1);
- }
-});
diff --git a/tools/cli/src/bin/dev.ts b/tools/cli/src/bin/dev.ts
deleted file mode 100644
index 8962d797e263b..0000000000000
--- a/tools/cli/src/bin/dev.ts
+++ /dev/null
@@ -1,146 +0,0 @@
-import { spawn } from 'node:child_process';
-import { existsSync } from 'node:fs';
-import { join } from 'node:path';
-
-import * as p from '@clack/prompts';
-import { config } from 'dotenv';
-import webpack from 'webpack';
-import WebpackDevServer from 'webpack-dev-server';
-
-import { getCwdFromDistribution, projectRoot } from '../config/cwd.cjs';
-import type { BuildFlags } from '../config/index.js';
-import { createWebpackConfig } from '../webpack/webpack.config.js';
-
-const flags: BuildFlags = {
- distribution:
- (process.env.DISTRIBUTION as BuildFlags['distribution']) ?? 'web',
- mode: 'development',
- static: false,
- channel: 'canary',
- coverage: process.env.COVERAGE === 'true',
-};
-
-const files = ['.env', '.env.local'];
-
-for (const file of files) {
- if (existsSync(join(projectRoot, file))) {
- config({
- path: join(projectRoot, file),
- });
- console.log(`${file} loaded`);
- break;
- }
-}
-
-const buildFlags = process.argv.includes('--static')
- ? { ...flags, static: true }
- : ((await p.group(
- {
- distribution: () =>
- p.select({
- message: 'Distribution',
- options: [
- {
- value: 'web',
- },
- {
- value: 'desktop',
- },
- {
- value: 'admin',
- },
- {
- value: 'mobile',
- },
- {
- value: 'ios',
- },
- ],
- initialValue: 'web',
- }),
- mode: () =>
- p.select({
- message: 'Mode',
- options: [
- {
- value: 'development',
- },
- {
- value: 'production',
- },
- ],
- initialValue: 'development',
- }),
- channel: () =>
- p.select({
- message: 'Channel',
- options: [
- {
- value: 'canary',
- },
- {
- value: 'beta',
- },
- {
- value: 'stable',
- },
- ],
- initialValue: 'canary',
- }),
- coverage: () =>
- p.confirm({
- message: 'Enable coverage',
- initialValue: process.env.COVERAGE === 'true',
- }),
- },
- {
- onCancel: () => {
- p.cancel('Operation cancelled.');
- process.exit(0);
- },
- }
- )) as BuildFlags);
-
-flags.distribution = buildFlags.distribution;
-flags.mode = buildFlags.mode;
-flags.channel = buildFlags.channel;
-flags.coverage = buildFlags.coverage;
-flags.static = buildFlags.static;
-flags.entry = undefined;
-
-const cwd = getCwdFromDistribution(flags.distribution);
-
-process.env.DISTRIBUTION = flags.distribution;
-
-if (flags.distribution === 'desktop') {
- flags.entry = {
- app: join(cwd, 'index.tsx'),
- shell: join(cwd, 'shell/index.tsx'),
- };
-}
-
-console.info(flags);
-
-if (!flags.static) {
- spawn('yarn', ['workspace', '@affine/i18n', 'dev'], {
- stdio: 'inherit',
- shell: true,
- });
-}
-
-try {
- // @ts-expect-error no types
- await import('@affine/templates/build-edgeless');
- const config = createWebpackConfig(cwd, flags);
- if (flags.static) {
- config.watch = false;
- }
- const compiler = webpack(config);
- // Start webpack
- const devServer = new WebpackDevServer(config.devServer, compiler);
-
- await devServer.start();
-} catch (error) {
- console.error('Error during build:', error);
- process.exit(1);
-}
diff --git a/tools/cli/src/config/cwd.cjs b/tools/cli/src/config/cwd.cjs
deleted file mode 100644
index b01b774455bcb..0000000000000
--- a/tools/cli/src/config/cwd.cjs
+++ /dev/null
@@ -1,38 +0,0 @@
-// @ts-check
-
-const { join } = require('node:path');
-
-const projectRoot = join(__dirname, '../../../..');
-
-module.exports.projectRoot = projectRoot;
-
-/**
- *
- * @param {string | undefined} distribution
- * @returns string
- */
-module.exports.getCwdFromDistribution = function getCwdFromDistribution(
- distribution
-) {
- switch (distribution) {
- case 'web':
- case undefined:
- case null:
- return join(projectRoot, 'packages/frontend/apps/web');
- case 'desktop':
- return join(projectRoot, 'packages/frontend/apps/electron/renderer');
- case 'admin':
- return join(projectRoot, 'packages/frontend/admin');
- case 'mobile':
- return join(projectRoot, 'packages/frontend/apps/mobile');
- case 'ios':
- return join(projectRoot, 'packages/frontend/apps/ios');
- case 'android':
- return join(projectRoot, 'packages/frontend/apps/android');
- default: {
- throw new Error(
- 'DISTRIBUTION must be one of web, desktop, admin, mobile'
- );
- }
- }
-};
diff --git a/tools/cli/src/config/index.ts b/tools/cli/src/config/index.ts
deleted file mode 100644
index f4452c5da070a..0000000000000
--- a/tools/cli/src/config/index.ts
+++ /dev/null
@@ -1,9 +0,0 @@
-export type BuildFlags = {
- distribution: 'web' | 'desktop' | 'admin' | 'mobile' | 'ios' | 'android';
- mode: 'development' | 'production';
- channel: 'stable' | 'beta' | 'canary' | 'internal';
- static: boolean;
- coverage?: boolean;
- localBlockSuite?: string;
- entry?: string | { [key: string]: string };
-};
diff --git a/tools/cli/src/util/infra.ts b/tools/cli/src/util/infra.ts
deleted file mode 100644
index 4e88c9ae8d816..0000000000000
--- a/tools/cli/src/util/infra.ts
+++ /dev/null
@@ -1,26 +0,0 @@
-import { spawn } from 'node:child_process';
-import { resolve } from 'node:path';
-
-import { build } from 'vite';
-
-import { projectRoot } from '../config/cwd.cjs';
-
-const infraFilePath = resolve(
- projectRoot,
- 'packages',
- 'infra',
- 'vite.config.ts'
-);
-export const buildInfra = async () => {
- await build({
- configFile: infraFilePath,
- });
-};
-
-export const watchInfra = async () => {
- spawn('vite', ['build', '--watch'], {
- cwd: resolve(projectRoot, 'packages/common/infra'),
- shell: true,
- stdio: 'inherit',
- });
-};
diff --git a/tools/cli/src/webpack/postcss.config.cjs b/tools/cli/src/webpack/postcss.config.cjs
deleted file mode 100644
index 9bc817f28a73f..0000000000000
--- a/tools/cli/src/webpack/postcss.config.cjs
+++ /dev/null
@@ -1,47 +0,0 @@
-const { join } = require('node:path');
-
-const cssnano = require('cssnano');
-const tailwindcss = require('tailwindcss');
-const autoprefixer = require('autoprefixer');
-
-const { getCwdFromDistribution } = require('../config/cwd.cjs');
-
-const projectCwd = getCwdFromDistribution(process.env.DISTRIBUTION);
-
-const twConfig = (function () {
- try {
- const config = require(`${projectCwd}/tailwind.config.js`);
- const { content } = config;
- if (Array.isArray(content)) {
- config.content = content.map(c =>
- c.startsWith(projectCwd) ? c : join(projectCwd, c)
- );
- }
- return config;
- } catch {
- return null;
- }
-})();
-
-module.exports = function (context) {
- const plugins = [
- cssnano({
- preset: [
- 'default',
- {
- convertValues: false,
- },
- ],
- }),
- ];
-
- if (twConfig) {
- plugins.push(tailwindcss(twConfig), autoprefixer());
- }
-
- return {
- from: context.from,
- plugins,
- to: context.to,
- };
-};
diff --git a/tools/cli/src/webpack/webpack.config.ts b/tools/cli/src/webpack/webpack.config.ts
deleted file mode 100644
index 290ce1fc386cc..0000000000000
--- a/tools/cli/src/webpack/webpack.config.ts
+++ /dev/null
@@ -1,183 +0,0 @@
-import { execSync } from 'node:child_process';
-import { readFileSync } from 'node:fs';
-import { join, resolve } from 'node:path';
-
-import type { BuildFlags } from '@affine/cli/config';
-import { Repository } from '@napi-rs/simple-git';
-import HTMLPlugin from 'html-webpack-plugin';
-import { once } from 'lodash-es';
-import type { Compiler } from 'webpack';
-import webpack from 'webpack';
-import { merge } from 'webpack-merge';
-
-import {
- createConfiguration,
- getPublicPath,
- rootPath,
- workspaceRoot,
-} from './config.js';
-import { getBuildConfig } from './runtime-config.js';
-
-const DESCRIPTION = `There can be more than Notion and Miro. AFFiNE is a next-gen knowledge base that brings planning, sorting and creating all together.`;
-
-const gitShortHash = once(() => {
- const { GITHUB_SHA } = process.env;
- if (GITHUB_SHA) {
- return GITHUB_SHA.substring(0, 9);
- }
- const repo = new Repository(workspaceRoot);
- const shortSha = repo.head().target()?.substring(0, 9);
- if (shortSha) {
- return shortSha;
- }
- const sha = execSync(`git rev-parse --short HEAD`, {
- encoding: 'utf-8',
- }).trim();
- return sha;
-});
-
-export function createWebpackConfig(cwd: string, flags: BuildFlags) {
- console.log('build flags', flags);
- const runtimeConfig = getBuildConfig(flags);
- console.log('BUILD_CONFIG', runtimeConfig);
- const config = createConfiguration(cwd, flags, runtimeConfig);
- const entry =
- typeof flags.entry === 'string' || !flags.entry
- ? {
- app: flags.entry ?? resolve(cwd, 'src/index.tsx'),
- }
- : flags.entry;
-
- const publicPath = getPublicPath(flags);
- const cdnOrigin = publicPath.startsWith('/')
- ? undefined
- : new URL(publicPath).origin;
-
- const globalErrorHandler = [
- 'js/global-error-handler.js',
- readFileSync(
- join(workspaceRoot, 'tools/cli/src/webpack/error-handler.js'),
- 'utf-8'
- ),
- ];
-
- const templateParams = {
- GIT_SHORT_SHA: gitShortHash(),
- DESCRIPTION,
- PRECONNECT: cdnOrigin
- ? ``
- : '',
- VIEWPORT_FIT:
- flags.distribution === 'mobile' ||
- flags.distribution === 'ios' ||
- flags.distribution === 'android'
- ? 'cover'
- : 'auto',
- };
-
- const createHTMLPlugins = (entryName: string) => {
- const htmlPluginOptions = {
- template: join(rootPath, 'webpack', 'template.html'),
- inject: 'body',
- filename: 'index.html',
- minify: false,
- templateParameters: templateParams,
- chunks: [entryName],
- } satisfies HTMLPlugin.Options;
-
- if (entryName === 'app') {
- return [
- {
- apply(compiler: Compiler) {
- compiler.hooks.compilation.tap(
- 'assets-manifest-plugin',
- compilation => {
- HTMLPlugin.getHooks(compilation).beforeAssetTagGeneration.tap(
- 'assets-manifest-plugin',
- arg => {
- if (
- flags.distribution !== 'desktop' &&
- !compilation.getAsset(globalErrorHandler[0])
- ) {
- compilation.emitAsset(
- globalErrorHandler[0],
- new webpack.sources.RawSource(globalErrorHandler[1])
- );
- arg.assets.js.unshift(
- arg.assets.publicPath + globalErrorHandler[0]
- );
- }
-
- if (!compilation.getAsset('assets-manifest.json')) {
- compilation.emitAsset(
- globalErrorHandler[0],
- new webpack.sources.RawSource(globalErrorHandler[1])
- );
- compilation.emitAsset(
- `assets-manifest.json`,
- new webpack.sources.RawSource(
- JSON.stringify(
- {
- ...arg.assets,
- js: arg.assets.js.map(file =>
- file.substring(arg.assets.publicPath.length)
- ),
- css: arg.assets.css.map(file =>
- file.substring(arg.assets.publicPath.length)
- ),
- gitHash: templateParams.GIT_SHORT_SHA,
- description: templateParams.DESCRIPTION,
- },
- null,
- 2
- )
- ),
- {
- immutable: false,
- }
- );
- }
-
- return arg;
- }
- );
- }
- );
- },
- },
- new HTMLPlugin({
- ...htmlPluginOptions,
- publicPath,
- meta: {
- 'env:publicPath': publicPath,
- },
- }),
- // selfhost html
- new HTMLPlugin({
- ...htmlPluginOptions,
- meta: {
- 'env:isSelfHosted': 'true',
- 'env:publicPath': '/',
- },
- filename: 'selfhost.html',
- templateParameters: {
- ...htmlPluginOptions.templateParameters,
- PRECONNECT: '',
- },
- }),
- ];
- } else {
- return [
- new HTMLPlugin({
- ...htmlPluginOptions,
- filename: `${entryName}.html`,
- }),
- ];
- }
- };
-
- return merge(config, {
- entry,
- plugins: Object.keys(entry).map(createHTMLPlugins).flat(),
- });
-}
diff --git a/tools/cli/tsconfig.json b/tools/cli/tsconfig.json
deleted file mode 100644
index 796d7ab24f54a..0000000000000
--- a/tools/cli/tsconfig.json
+++ /dev/null
@@ -1,12 +0,0 @@
-{
- "extends": "../../tsconfig.json",
- "compilerOptions": {
- "composite": true,
- "allowJs": true,
- "module": "ESNext",
- "moduleResolution": "Node",
- "outDir": "lib"
- },
- "include": ["src", "package.json"],
- "references": [{ "path": "../../packages/common/env" }]
-}
diff --git a/tools/commitlint/.commitlintrc.json b/tools/commitlint/.commitlintrc.json
index e3ef1467c514a..8be259393347f 100644
--- a/tools/commitlint/.commitlintrc.json
+++ b/tools/commitlint/.commitlintrc.json
@@ -14,19 +14,17 @@
"ios",
"android",
"docs",
- "storybook",
"component",
- "workspace",
"env",
"graphql",
- "cli",
"hooks",
"i18n",
"native",
"templates",
"debug",
"nbstore",
- "infra"
+ "infra",
+ "tools"
]
]
}
diff --git a/tools/scripts/README.md b/tools/scripts/README.md
new file mode 100644
index 0000000000000..f5fd4153ec166
--- /dev/null
+++ b/tools/scripts/README.md
@@ -0,0 +1,74 @@
+# AFFiNE Monorepo scripts
+
+## Start
+
+```bash
+yarn affine -h
+```
+
+### Run build command defined in package.json
+
+```bash
+yarn affine i18n build
+# or
+yarn build -p i18n
+```
+
+### Run dev command defined in package.json
+
+```bash
+yarn affine web dev
+# or
+yarn dev -p i18n
+```
+
+### Clean
+
+```bash
+yarn affine clean --ts --dist --rust
+# clean node_modules
+yarn affine clean --node-modules
+```
+
+## Tricks
+
+### Short your key presses
+
+```bash
+# af is also available for running the scripts
+yarn af web build
+```
+
+#### by custom shell script
+
+> personally, I use 'af', and only demoed in macos.
+
+create file `af` in the root of AFFiNE project with the following content
+
+```bash
+#!/usr/bin/env sh
+./tools/scripts/bin/runner.js affine.ts $@
+```
+
+and give it executable permission
+
+```bash
+chmod a+x ./af
+
+# now you can run scripts with simply
+./af web build
+```
+
+if you want to go further, but for vscode(or other forks) only, add the following to your `.vscode/settings.json`
+
+```json
+"terminal.integrated.env.osx": {
+ "PATH": ".:$PATH"
+}
+```
+
+restart all the integrated terminals and now you get:
+
+```bash
+af web build
+```
diff --git a/tools/scripts/bin/runner.js b/tools/scripts/bin/runner.js
new file mode 100755
index 0000000000000..8f10896225fea
--- /dev/null
+++ b/tools/scripts/bin/runner.js
@@ -0,0 +1,68 @@
+#!/usr/bin/env node
+import { spawn } from 'node:child_process';
+import { existsSync } from 'node:fs';
+import { join } from 'node:path';
+import { fileURLToPath } from 'node:url';
+
+const scriptsFolder = join(fileURLToPath(import.meta.url), '..', '..');
+const scriptsSrcFolder = join(scriptsFolder, 'src');
+const projectRoot = join(scriptsFolder, '..', '..');
+const loader = join(scriptsFolder, 'loader.js');
+
+const [node, _self, file, ...options] = process.argv;
+
+if (!file) {
+ console.error(`Please provide a file to run, e.g. 'run src/index.{js/ts}'`);
+ process.exit(1);
+}
+
+const fileLocationCandidates = new Set([
+ process.cwd(),
+ scriptsSrcFolder,
+ projectRoot,
+]);
+const lookups = [];
+
+/**
+ * @type {string | undefined}
+ */
+let scriptLocation;
+for (const location of fileLocationCandidates) {
+ const fileCandidates = [file, `${file}.js`, `${file}.ts`];
+ for (const candidate of fileCandidates) {
+ const candidateLocation = join(location, candidate);
+ if (existsSync(candidateLocation)) {
+ scriptLocation = candidateLocation;
+ break;
+ }
+ lookups.push(candidateLocation);
+ }
+}
+
+if (!scriptLocation) {
+ console.error(
+ `File ${file} not found, please make sure the first parameter passed to 'run' script is a valid js or ts file.`
+ );
+ console.error(`Searched locations: `);
+ lookups.forEach(location => {
+ console.error(` - ${location}`);
+ });
+ process.exit(1);
+}
+
+const nodeOptions = [];
+
+if (
+ scriptLocation.endsWith('.ts') ||
+ scriptLocation.startsWith(scriptsFolder)
+) {
+ nodeOptions.unshift(`--import=${loader}`);
+}
+
+nodeOptions.unshift('--experimental-specifier-resolution=node');
+
+spawn(node, [...nodeOptions, scriptLocation, ...options], {
+ stdio: 'inherit',
+}).on('exit', code => {
+ process.exit(code);
+});
diff --git a/tools/scripts/loader.js b/tools/scripts/loader.js
new file mode 100644
index 0000000000000..57272fa69dbd0
--- /dev/null
+++ b/tools/scripts/loader.js
@@ -0,0 +1,4 @@
+import { register } from 'node:module';
+import { pathToFileURL } from 'node:url';
+
+register('ts-node/esm/transpile-only.mjs', pathToFileURL('./'));
diff --git a/tools/cli/package.json b/tools/scripts/package.json
similarity index 69%
rename from tools/cli/package.json
rename to tools/scripts/package.json
index ee11989048b11..9f3196db5e47d 100644
--- a/tools/cli/package.json
+++ b/tools/scripts/package.json
@@ -1,27 +1,34 @@
{
- "name": "@affine/cli",
+ "name": "@affine-tools/scripts",
+ "version": "0.0.1",
"type": "module",
"private": true,
+ "bin": {
+ "r": "./bin/runner.js"
+ },
+ "exports": {
+ "./loader": "./loader.js"
+ },
+ "scripts": {
+ "affine": "r ./src/affine.ts"
+ },
"devDependencies": {
- "@affine/env": "workspace:*",
- "@affine/templates": "workspace:*",
+ "@affine-tools/utils": "workspace:*",
"@aws-sdk/client-s3": "^3.709.0",
- "@blocksuite/affine": "0.19.3",
- "@clack/core": "^0.3.5",
- "@clack/prompts": "^0.8.2",
- "@magic-works/i18n-codegen": "^0.6.1",
"@napi-rs/simple-git": "^0.1.19",
"@perfsee/webpack": "^1.13.0",
"@pmmmwh/react-refresh-webpack-plugin": "^0.5.15",
"@sentry/webpack-plugin": "^2.22.7",
+ "@types/lodash-es": "^4.17.12",
"@types/mime-types": "^2.1.4",
+ "@types/node": "^20.17.10",
"@types/webpack-env": "^1.18.5",
"@vanilla-extract/webpack-plugin": "^2.3.15",
"autoprefixer": "^10.4.20",
+ "clipanion": "^3.2.1",
"copy-webpack-plugin": "^12.0.2",
"css-loader": "^7.1.2",
"cssnano": "^7.0.6",
- "dotenv": "^16.4.7",
"html-webpack-plugin": "^5.6.3",
"lodash-es": "^4.17.21",
"mime-types": "^2.1.35",
@@ -35,14 +42,10 @@
"tailwindcss": "^3.4.16",
"terser-webpack-plugin": "^5.3.10",
"ts-node": "^10.9.2",
- "vite": "^6.0.3",
+ "typanion": "^3.14.0",
+ "typescript": "^5.5.4",
"webpack": "^5.97.1",
"webpack-dev-server": "^5.2.0",
"webpack-merge": "^6.0.1"
- },
- "scripts": {
- "bundle": "node --loader ts-node/esm/transpile-only.mjs ./src/bin/build.ts",
- "dev": "node --loader ts-node/esm/transpile-only.mjs ./src/bin/dev.ts"
- },
- "version": "0.18.0"
+ }
}
diff --git a/tools/scripts/src/affine.ts b/tools/scripts/src/affine.ts
new file mode 100644
index 0000000000000..d69acfaabe6eb
--- /dev/null
+++ b/tools/scripts/src/affine.ts
@@ -0,0 +1,32 @@
+import { Workspace } from '@affine-tools/utils/workspace';
+import { Cli } from 'clipanion';
+
+import { BuildCommand } from './build';
+import { BundleCommand } from './bundle';
+import { CleanCommand } from './clean';
+import { CodegenCommand } from './codegen';
+import type { CliContext } from './context';
+import { DevCommand } from './dev';
+import { RunCommand } from './run';
+
+const cli = new Cli({
+ binaryName: 'affine',
+ binaryVersion: '0.0.0',
+ binaryLabel: 'AFFiNE Monorepo Tools',
+ enableColors: true,
+ enableCapture: true,
+});
+
+cli.register(RunCommand);
+cli.register(CodegenCommand);
+cli.register(CleanCommand);
+cli.register(BuildCommand);
+cli.register(DevCommand);
+cli.register(BundleCommand);
+
+await cli.runExit(process.argv.slice(2), {
+ workspace: new Workspace(),
+ stdin: process.stdin,
+ stdout: process.stdout,
+ stderr: process.stderr,
+});
diff --git a/tools/scripts/src/build.ts b/tools/scripts/src/build.ts
new file mode 100644
index 0000000000000..94337b55d3ad3
--- /dev/null
+++ b/tools/scripts/src/build.ts
@@ -0,0 +1,15 @@
+import { PackageCommand } from './command';
+
+export class BuildCommand extends PackageCommand {
+ static override paths = [['build'], ['b']];
+
+ async execute() {
+ const args = ['affine build', this.package];
+
+ if (this.deps) {
+ args.push('--deps');
+ }
+
+ await this.cli.run(args);
+ }
+}
diff --git a/tools/scripts/src/bundle.ts b/tools/scripts/src/bundle.ts
new file mode 100644
index 0000000000000..933000c03f957
--- /dev/null
+++ b/tools/scripts/src/bundle.ts
@@ -0,0 +1,93 @@
+import webpack, { type Compiler, type Configuration } from 'webpack';
+import WebpackDevServer from 'webpack-dev-server';
+import { merge } from 'webpack-merge';
+
+import { Option, PackageCommand } from './command';
+import { createWebpackConfig } from './webpack';
+
+function getChannel() {
+ const channel = process.env.BUILD_TYPE ?? 'canary';
+ switch (channel) {
+ case 'canary':
+ case 'beta':
+ case 'stable':
+ case 'internal':
+ return channel;
+ default: {
+ throw new Error(
+ `BUILD_TYPE must be one of canary, beta, stable, internal, received [${channel}]`
+ );
+ }
+ }
+}
+
+export class BundleCommand extends PackageCommand {
+ static override paths = [['bundle'], ['webpack'], ['pack'], ['bun']];
+
+ // bundle is not able to run with deps
+ override deps = false;
+
+ dev = Option.Boolean('--dev,-d', false, {
+ description: 'Run in Development mode',
+ });
+
+ async execute() {
+ this.logger.info(`Packing package ${this.package}...`);
+
+ const config = await this.getConfig();
+
+ const compiler = webpack(config);
+
+ if (this.dev) {
+ await this.start(compiler, config.devServer);
+ } else {
+ await this.build(compiler);
+ }
+ }
+
+ async getConfig() {
+ let config = createWebpackConfig(this.workspace.getPackage(this.package), {
+ mode: this.dev ? 'development' : 'production',
+ channel: getChannel(),
+ });
+
+ let configOverride: Configuration | undefined;
+ const overrideConfigPath = this.workspace
+ .getPackage(this.package)
+ .join('webpack.config.ts');
+
+ if (overrideConfigPath.isFile()) {
+ const override = await import(overrideConfigPath.value);
+ configOverride = override.config ?? override.default;
+ }
+
+ if (configOverride) {
+ config = merge(config, configOverride);
+ }
+
+ return config;
+ }
+
+ async start(compiler: Compiler, config: Configuration['devServer']) {
+ const devServer = new WebpackDevServer(config, compiler);
+
+ await devServer.start();
+ }
+
+ async build(compiler: Compiler) {
+ compiler.run((error, stats) => {
+ if (error) {
+ console.error(error);
+ process.exit(1);
+ }
+ if (stats) {
+ if (stats.hasErrors()) {
+ console.error(stats.toString('errors-only'));
+ process.exit(1);
+ } else {
+ console.log(stats.toString('minimal'));
+ }
+ }
+ });
+ }
+}
diff --git a/tools/scripts/src/clean.ts b/tools/scripts/src/clean.ts
new file mode 100644
index 0000000000000..b963907032cfb
--- /dev/null
+++ b/tools/scripts/src/clean.ts
@@ -0,0 +1,84 @@
+import { rmSync } from 'node:fs';
+
+import { exec } from '@affine-tools/utils/process';
+
+import { Command, Option } from './command';
+
+export class CleanCommand extends Command {
+ static override paths = [['clean']];
+
+ cleanTsBuild = Option.Boolean('--ts', false);
+ cleanDist = Option.Boolean('--dist', false);
+ cleanRustTarget = Option.Boolean('--rust', false);
+ cleanNodeModules = Option.Boolean('--node-modules', false);
+ all = Option.Boolean('--all,-a', false);
+
+ async execute() {
+ this.logger.info('Cleaning Workspace...');
+ if (this.all || this.cleanNodeModules) {
+ this.doCleanNodeModules();
+ }
+
+ if (this.all || this.cleanDist) {
+ this.doCleanDist();
+ }
+
+ if (this.all || this.cleanRustTarget) {
+ this.doCleanRust();
+ }
+
+ if (this.all || this.cleanTsBuild) {
+ this.doCleanTs();
+ }
+ }
+
+ doCleanNodeModules() {
+ this.logger.info('Cleaning node_modules...');
+
+ const rootNodeModules = this.workspace.join('node_modules');
+ if (rootNodeModules.isDirectory()) {
+ this.logger.info(`Cleaning ${rootNodeModules}`);
+ rmSync(rootNodeModules.value, { recursive: true });
+ }
+
+ this.workspace.forEach(pkg => {
+ const nodeModules = pkg.nodeModulesPath;
+ if (nodeModules.isDirectory()) {
+ this.logger.info(`Cleaning ${nodeModules}`);
+ rmSync(nodeModules.value, { recursive: true });
+ }
+ });
+
+ this.logger.info('node_modules cleaned');
+ }
+
+ doCleanTs() {
+ this.logger.info('Cleaning ts build outputs...');
+
+ this.workspace.forEach(pkg => {
+ if (pkg.tsbuildPath.isDirectory()) {
+ this.logger.info(`Cleaning ${pkg.tsbuildPath}`);
+ rmSync(pkg.tsbuildPath.value, { recursive: true });
+ }
+ });
+
+ this.logger.info('ts build outputs cleaned');
+ }
+
+ doCleanDist() {
+ this.logger.info('Cleaning dist...');
+
+ this.workspace.forEach(pkg => {
+ if (pkg.distPath.isDirectory()) {
+ this.logger.info(`Cleaning ${pkg.distPath}`);
+ rmSync(pkg.distPath.value, { recursive: true });
+ }
+ });
+
+ this.logger.info('dist cleaned');
+ }
+
+ doCleanRust() {
+ exec('', 'cargo clean');
+ }
+}
diff --git a/tools/scripts/src/codegen.ts b/tools/scripts/src/codegen.ts
new file mode 100755
index 0000000000000..35a7946d3fe80
--- /dev/null
+++ b/tools/scripts/src/codegen.ts
@@ -0,0 +1,35 @@
+import { writeFileSync } from 'node:fs';
+
+import type { Path } from '@affine-tools/utils/path';
+
+import { Command } from './command';
+
+export class CodegenCommand extends Command {
+ static override paths = [['codegen'], ['gg'], ['g']];
+
+ async execute() {
+ this.logger.info('Generating Workspace configs');
+ this.generateWorkspaceFiles();
+ this.logger.info('Workspace configs generated');
+ }
+
+ generateWorkspaceFiles() {
+ const filesToGenerate: [Path, () => string][] = [
+ [
+ this.workspace.join('tsconfig.project.json'),
+ this.workspace.genProjectTsConfig,
+ ],
+ [
+ this.workspace
+ .getPackage('@affine-tools/utils')
+ .join('src/workspace.gen.ts'),
+ this.workspace.genWorkspaceInfo,
+ ],
+ ];
+
+ for (const [path, content] of filesToGenerate) {
+ this.logger.info(`Output: ${path}`);
+ writeFileSync(path.value, content.apply(this.workspace));
+ }
+ }
+}
diff --git a/tools/scripts/src/command.ts b/tools/scripts/src/command.ts
new file mode 100644
index 0000000000000..ab5b54da14c06
--- /dev/null
+++ b/tools/scripts/src/command.ts
@@ -0,0 +1,71 @@
+import { AliasToPackage } from '@affine-tools/utils/distribution';
+import { Logger } from '@affine-tools/utils/logger';
+import { type PackageName, Workspace } from '@affine-tools/utils/workspace';
+import { Command as BaseCommand, Option } from 'clipanion';
+import * as t from 'typanion';
+
+import type { CliContext } from './context';
+
+export abstract class Command extends BaseCommand {
+ get logger() {
+ // @ts-expect-error hack: Get the command name
+ return new Logger(this.constructor.paths[0][0]);
+ }
+
+ get workspace() {
+ return this.context.workspace;
+ }
+}
+
+export abstract class PackageCommand extends Command {
+ protected availablePackageNameArgs = (
+ Workspace.PackageNames as string[]
+ ).concat(Array.from(AliasToPackage.keys()));
+ protected packageNameValidator = t.isOneOf(
+ this.availablePackageNameArgs.map(k => t.isLiteral(k))
+ );
+
+ protected packageNameOrAlias = Option.String('--package,-p', {
+ required: true,
+ validator: this.packageNameValidator,
+ description: 'The package name or alias to be run with',
+ });
+
+ get package(): PackageName {
+ return (
+ AliasToPackage.get(this.packageNameOrAlias as any) ??
+ (this.packageNameOrAlias as PackageName)
+ );
+ }
+
+ deps = Option.Boolean('--deps', false, {
+ description:
+ 'Execute the same command in workspace dependencies, if defined.',
+ });
+}
+
+export abstract class PackagesCommand extends Command {
+ protected availablePackageNameArgs = (
+ Workspace.PackageNames as string[]
+ ).concat(Array.from(AliasToPackage.keys()));
+ protected packageNameValidator = t.isOneOf(
+ this.availablePackageNameArgs.map(k => t.isLiteral(k))
+ );
+
+ protected packageNamesOrAliases = Option.Array('--package,-p', {
+ required: true,
+ validator: t.isArray(this.packageNameValidator),
+ });
+ get packages() {
+ return this.packageNamesOrAliases.map(
+ name => AliasToPackage.get(name as any) ?? name
+ );
+ }
+
+ deps = Option.Boolean('--deps', false, {
+ description:
+ 'Execute the same command in workspace dependencies, if defined.',
+ });
+}
+
+export { Option };
diff --git a/tools/scripts/src/context.ts b/tools/scripts/src/context.ts
new file mode 100644
index 0000000000000..bc97d15a67328
--- /dev/null
+++ b/tools/scripts/src/context.ts
@@ -0,0 +1,6 @@
+import type { Workspace } from '@affine-tools/utils/workspace';
+import type { BaseContext } from 'clipanion';
+
+export interface CliContext extends BaseContext {
+ workspace: Workspace;
+}
diff --git a/tools/scripts/src/dev.ts b/tools/scripts/src/dev.ts
new file mode 100644
index 0000000000000..efa3c3028c937
--- /dev/null
+++ b/tools/scripts/src/dev.ts
@@ -0,0 +1,15 @@
+import { PackageCommand } from './command';
+
+export class DevCommand extends PackageCommand {
+ static override paths = [['dev'], ['d']];
+
+ async execute() {
+ const args = [this.package, 'dev'];
+
+ if (this.deps) {
+ args.push('--deps');
+ }
+
+ await this.cli.run(args);
+ }
+}
diff --git a/tools/scripts/src/run.ts b/tools/scripts/src/run.ts
new file mode 100644
index 0000000000000..dc747d8794c7e
--- /dev/null
+++ b/tools/scripts/src/run.ts
@@ -0,0 +1,111 @@
+import { Path } from '@affine-tools/utils/path';
+import { execAsync } from '@affine-tools/utils/process';
+import type { PackageName } from '@affine-tools/utils/workspace';
+
+import { Option, PackageCommand } from './command';
+
+interface RunScriptOptions {
+ ignoreMissingScript?: boolean;
+ includeDependencies?: boolean;
+ waitDependencies?: boolean;
+}
+
+const currentDir = Path.dir(import.meta.url);
+
+export class RunCommand extends PackageCommand {
+ static override paths = [[], ['run'], ['r']];
+
+ static override usage = PackageCommand.Usage({
+ description: 'AFFiNE Monorepo scripts',
+ details: `
+ \`affine web