mirror of
https://github.com/LukeHagar/vercel.git
synced 2025-12-12 04:22:14 +00:00
Compare commits
33 Commits
api@0.0.1
...
@vercel/bu
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
82231058da | ||
|
|
61227bf7e3 | ||
|
|
6aa0aa4e65 | ||
|
|
caaba0d685 | ||
|
|
335fd70a68 | ||
|
|
c3c54d6e69 | ||
|
|
43048a0dd8 | ||
|
|
60c75fd76c | ||
|
|
bd1319d7a3 | ||
|
|
1138f7e3d1 | ||
|
|
bb95cb9225 | ||
|
|
0048eb999e | ||
|
|
1b4de4a986 | ||
|
|
c9ad4084ee | ||
|
|
98ee6e4728 | ||
|
|
d7db0c5794 | ||
|
|
e90e922ee8 | ||
|
|
adb9ac87ce | ||
|
|
e43191b186 | ||
|
|
a962e84409 | ||
|
|
7593e219fd | ||
|
|
ec95066689 | ||
|
|
42a71416b9 | ||
|
|
80af30ce60 | ||
|
|
74a15b5f32 | ||
|
|
c0416f7e95 | ||
|
|
fc1e13c099 | ||
|
|
0e9ec194a3 | ||
|
|
5609a1187b | ||
|
|
cff72e3129 | ||
|
|
d3c84e5d2a | ||
|
|
9e3827c785 | ||
|
|
fb6d77afac |
4
.github/CODEOWNERS
vendored
4
.github/CODEOWNERS
vendored
@@ -5,12 +5,12 @@
|
|||||||
* @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek
|
* @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek
|
||||||
/.github/workflows @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @ijjk
|
/.github/workflows @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @ijjk
|
||||||
/packages/fs-detectors @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @agadzik @chloetedder
|
/packages/fs-detectors @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @agadzik @chloetedder
|
||||||
/packages/next @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @ijjk
|
/packages/next @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @ijjk @ztanner
|
||||||
/packages/routing-utils @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @ijjk
|
/packages/routing-utils @TooTallNate @EndangeredMassa @styfle @cb1kenobi @Ethan-Arrowood @trek @ijjk
|
||||||
/packages/edge @vercel/compute
|
/packages/edge @vercel/compute
|
||||||
/examples @leerob
|
/examples @leerob
|
||||||
/examples/create-react-app @Timer
|
/examples/create-react-app @Timer
|
||||||
/examples/nextjs @timneutkens @ijjk @styfle
|
/examples/nextjs @timneutkens @ijjk @styfle @ztanner
|
||||||
/examples/hugo @styfle
|
/examples/hugo @styfle
|
||||||
/examples/jekyll @styfle
|
/examples/jekyll @styfle
|
||||||
/examples/zola @styfle
|
/examples/zola @styfle
|
||||||
|
|||||||
7
examples/CHANGELOG.md
vendored
Normal file
7
examples/CHANGELOG.md
vendored
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
# examples
|
||||||
|
|
||||||
|
## null
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- update examples to use at least node@16 ([#10395](https://github.com/vercel/vercel/pull/10395))
|
||||||
3
examples/package.json
vendored
3
examples/package.json
vendored
@@ -10,5 +10,6 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/jest": "27.4.1",
|
"@types/jest": "27.4.1",
|
||||||
"@vercel/frameworks": "2.0.1"
|
"@vercel/frameworks": "2.0.1"
|
||||||
}
|
},
|
||||||
|
"version": null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,6 +30,6 @@
|
|||||||
"wct-browser-legacy": "^1.0.0"
|
"wct-browser-legacy": "^1.0.0"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=8.0"
|
"node": ">=16.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,7 @@
|
|||||||
"extends": "@redwoodjs/eslint-config"
|
"extends": "@redwoodjs/eslint-config"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=12",
|
"node": "16.x",
|
||||||
"yarn": ">=1.15"
|
"yarn": ">=1.15"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,6 +27,6 @@
|
|||||||
"typescript": "^5.1.3"
|
"typescript": "^5.1.3"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=14"
|
"node": ">=16"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -46,6 +46,6 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=8.0.0 <17"
|
"node": "16.x"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
"build": "vuepress build src"
|
"build": "vuepress build src"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=14.0.0 <18.0.0"
|
"node": ">=16.0.0 <18.0.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"vuepress": "1.9.7"
|
"vuepress": "1.9.7"
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
"dist"
|
"dist"
|
||||||
],
|
],
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsc",
|
"build": "node ../../utils/build.mjs",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
||||||
"test-unit": "pnpm test tests/unit"
|
"test-unit": "pnpm test tests/unit"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
{
|
{
|
||||||
"extends": "@vercel-internals/tsconfig",
|
"extends": "@vercel-internals/tsconfig",
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"outDir": "dist"
|
"outDir": "dist",
|
||||||
|
"declaration": true,
|
||||||
|
"emitDeclarationOnly": true
|
||||||
},
|
},
|
||||||
"include": ["src/**/*"]
|
"include": ["src/**/*"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,19 @@
|
|||||||
# @vercel-internals/types
|
# @vercel-internals/types
|
||||||
|
|
||||||
|
## 1.0.9
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`5609a1187`](https://github.com/vercel/vercel/commit/5609a1187be9d6cf8d5f16825690c5ea72f17dc5), [`1b4de4a98`](https://github.com/vercel/vercel/commit/1b4de4a986f7a612aac834ebae3ec7bb9e9b8cf8)]:
|
||||||
|
- @vercel/build-utils@7.1.1
|
||||||
|
|
||||||
|
## 1.0.8
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`9e3827c78`](https://github.com/vercel/vercel/commit/9e3827c785e1bc45f2bed421132167381481770f)]:
|
||||||
|
- @vercel/build-utils@7.1.0
|
||||||
|
|
||||||
## 1.0.7
|
## 1.0.7
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"private": true,
|
"private": true,
|
||||||
"name": "@vercel-internals/types",
|
"name": "@vercel-internals/types",
|
||||||
"version": "1.0.7",
|
"version": "1.0.9",
|
||||||
"types": "index.d.ts",
|
"types": "index.d.ts",
|
||||||
"main": "index.d.ts",
|
"main": "index.d.ts",
|
||||||
"files": [
|
"files": [
|
||||||
@@ -10,7 +10,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/node": "14.14.31",
|
"@types/node": "14.14.31",
|
||||||
"@vercel-internals/constants": "1.0.4",
|
"@vercel-internals/constants": "1.0.4",
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/routing-utils": "3.0.0"
|
"@vercel/routing-utils": "3.0.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|||||||
@@ -16,6 +16,7 @@
|
|||||||
"buffer-replace": "1.0.0",
|
"buffer-replace": "1.0.0",
|
||||||
"create-svelte": "2.0.1",
|
"create-svelte": "2.0.1",
|
||||||
"dot": "1.1.3",
|
"dot": "1.1.3",
|
||||||
|
"esbuild": "0.19.2",
|
||||||
"eslint": "8.14.0",
|
"eslint": "8.14.0",
|
||||||
"eslint-config-prettier": "8.5.0",
|
"eslint-config-prettier": "8.5.0",
|
||||||
"eslint-plugin-jest": "26.1.5",
|
"eslint-plugin-jest": "26.1.5",
|
||||||
@@ -32,7 +33,7 @@
|
|||||||
"source-map-support": "0.5.12",
|
"source-map-support": "0.5.12",
|
||||||
"ts-eager": "2.0.2",
|
"ts-eager": "2.0.2",
|
||||||
"ts-jest": "29.1.0",
|
"ts-jest": "29.1.0",
|
||||||
"turbo": "1.10.12",
|
"turbo": "1.10.13",
|
||||||
"typescript": "4.9.5"
|
"typescript": "4.9.5"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -49,7 +50,7 @@
|
|||||||
"prepare": "husky install",
|
"prepare": "husky install",
|
||||||
"pack": "cd utils && node -r ts-eager/register ./pack.ts",
|
"pack": "cd utils && node -r ts-eager/register ./pack.ts",
|
||||||
"ci:version": "changeset version && pnpm install --no-frozen-lockfile",
|
"ci:version": "changeset version && pnpm install --no-frozen-lockfile",
|
||||||
"ci:publish": "pnpm publish -r && changeset tag"
|
"ci:publish": "pnpm publish -r && node utils/update-canary-tags.mjs && changeset tag"
|
||||||
},
|
},
|
||||||
"lint-staged": {
|
"lint-staged": {
|
||||||
"./{*,{api,packages,test,utils}/**/*}.{js,ts}": [
|
"./{*,{api,packages,test,utils}/**/*}.{js,ts}": [
|
||||||
|
|||||||
@@ -1,5 +1,19 @@
|
|||||||
# @vercel/build-utils
|
# @vercel/build-utils
|
||||||
|
|
||||||
|
## 7.1.1
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- add descriptions to NodeVersion properties ([#10403](https://github.com/vercel/vercel/pull/10403))
|
||||||
|
|
||||||
|
- Updated semver dependency ([#10411](https://github.com/vercel/vercel/pull/10411))
|
||||||
|
|
||||||
|
## 7.1.0
|
||||||
|
|
||||||
|
### Minor Changes
|
||||||
|
|
||||||
|
- Support serverless function architecture ([#10392](https://github.com/vercel/vercel/pull/10392))
|
||||||
|
|
||||||
## 7.0.0
|
## 7.0.0
|
||||||
|
|
||||||
### Major Changes
|
### Major Changes
|
||||||
|
|||||||
@@ -1,30 +0,0 @@
|
|||||||
#!/usr/bin/env node
|
|
||||||
const fs = require('fs-extra');
|
|
||||||
const execa = require('execa');
|
|
||||||
const { join } = require('path');
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
const outDir = join(__dirname, 'dist');
|
|
||||||
|
|
||||||
// Start fresh
|
|
||||||
await fs.remove(outDir);
|
|
||||||
|
|
||||||
// Compile TypeScript
|
|
||||||
await execa('tsc', [], { stdio: 'inherit' });
|
|
||||||
|
|
||||||
// Run `ncc`
|
|
||||||
const mainDir = join(outDir, 'main');
|
|
||||||
await execa('ncc', ['build', 'src/index.ts', '-o', mainDir], {
|
|
||||||
stdio: 'inherit',
|
|
||||||
});
|
|
||||||
// Move compiled ncc file to out dir
|
|
||||||
await fs.rename(join(mainDir, 'index.js'), join(outDir, 'index.js'));
|
|
||||||
|
|
||||||
// Delete leftover "main" dir
|
|
||||||
await fs.remove(mainDir);
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(err => {
|
|
||||||
console.error(err);
|
|
||||||
process.exit(1);
|
|
||||||
});
|
|
||||||
3
packages/build-utils/build.mjs
Normal file
3
packages/build-utils/build.mjs
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
import { tsc, esbuild } from '../../utils/build.mjs';
|
||||||
|
|
||||||
|
await Promise.all([tsc(), esbuild().then(() => esbuild({ bundle: true }))]);
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vercel/build-utils",
|
"name": "@vercel/build-utils",
|
||||||
"version": "7.0.0",
|
"version": "7.1.1",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"main": "./dist/index.js",
|
"main": "./dist/index.js",
|
||||||
"types": "./dist/index.d.js",
|
"types": "./dist/index.d.js",
|
||||||
@@ -11,7 +11,7 @@
|
|||||||
"directory": "packages/now-build-utils"
|
"directory": "packages/now-build-utils"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "node build",
|
"build": "node build.mjs",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
||||||
"test-unit": "pnpm test test/unit.*test.*",
|
"test-unit": "pnpm test test/unit.*test.*",
|
||||||
"test-e2e": "pnpm test test/integration.test.ts"
|
"test-e2e": "pnpm test test/integration.test.ts"
|
||||||
@@ -49,7 +49,7 @@
|
|||||||
"minimatch": "3.1.2",
|
"minimatch": "3.1.2",
|
||||||
"multistream": "2.1.1",
|
"multistream": "2.1.1",
|
||||||
"node-fetch": "2.6.7",
|
"node-fetch": "2.6.7",
|
||||||
"semver": "6.1.1",
|
"semver": "6.3.1",
|
||||||
"typescript": "4.9.5",
|
"typescript": "4.9.5",
|
||||||
"yazl": "2.5.1"
|
"yazl": "2.5.1"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,9 +13,12 @@ interface Environment {
|
|||||||
|
|
||||||
export type LambdaOptions = LambdaOptionsWithFiles | LambdaOptionsWithZipBuffer;
|
export type LambdaOptions = LambdaOptionsWithFiles | LambdaOptionsWithZipBuffer;
|
||||||
|
|
||||||
|
export type LambdaArchitecture = 'x86_64' | 'arm64';
|
||||||
|
|
||||||
export interface LambdaOptionsBase {
|
export interface LambdaOptionsBase {
|
||||||
handler: string;
|
handler: string;
|
||||||
runtime: string;
|
runtime: string;
|
||||||
|
architecture?: LambdaArchitecture;
|
||||||
memory?: number;
|
memory?: number;
|
||||||
maxDuration?: number;
|
maxDuration?: number;
|
||||||
environment?: Environment;
|
environment?: Environment;
|
||||||
@@ -62,6 +65,7 @@ export class Lambda {
|
|||||||
files?: Files;
|
files?: Files;
|
||||||
handler: string;
|
handler: string;
|
||||||
runtime: string;
|
runtime: string;
|
||||||
|
architecture?: LambdaArchitecture;
|
||||||
memory?: number;
|
memory?: number;
|
||||||
maxDuration?: number;
|
maxDuration?: number;
|
||||||
environment: Environment;
|
environment: Environment;
|
||||||
@@ -81,6 +85,7 @@ export class Lambda {
|
|||||||
handler,
|
handler,
|
||||||
runtime,
|
runtime,
|
||||||
maxDuration,
|
maxDuration,
|
||||||
|
architecture,
|
||||||
memory,
|
memory,
|
||||||
environment = {},
|
environment = {},
|
||||||
allowQuery,
|
allowQuery,
|
||||||
@@ -102,6 +107,13 @@ export class Lambda {
|
|||||||
assert(typeof runtime === 'string', '"runtime" is not a string');
|
assert(typeof runtime === 'string', '"runtime" is not a string');
|
||||||
assert(typeof environment === 'object', '"environment" is not an object');
|
assert(typeof environment === 'object', '"environment" is not an object');
|
||||||
|
|
||||||
|
if (architecture !== undefined) {
|
||||||
|
assert(
|
||||||
|
architecture === 'x86_64' || architecture === 'arm64',
|
||||||
|
'"architecture" must be either "x86_64" or "arm64"'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (memory !== undefined) {
|
if (memory !== undefined) {
|
||||||
assert(typeof memory === 'number', '"memory" is not a number');
|
assert(typeof memory === 'number', '"memory" is not a number');
|
||||||
}
|
}
|
||||||
@@ -159,6 +171,7 @@ export class Lambda {
|
|||||||
this.files = 'files' in opts ? opts.files : undefined;
|
this.files = 'files' in opts ? opts.files : undefined;
|
||||||
this.handler = handler;
|
this.handler = handler;
|
||||||
this.runtime = runtime;
|
this.runtime = runtime;
|
||||||
|
this.architecture = architecture;
|
||||||
this.memory = memory;
|
this.memory = memory;
|
||||||
this.maxDuration = maxDuration;
|
this.maxDuration = maxDuration;
|
||||||
this.environment = environment;
|
this.environment = environment;
|
||||||
|
|||||||
@@ -300,9 +300,13 @@ export interface PackageJson {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface NodeVersion {
|
export interface NodeVersion {
|
||||||
|
/** major version number: 18 */
|
||||||
major: number;
|
major: number;
|
||||||
|
/** major version range: "18.x" */
|
||||||
range: string;
|
range: string;
|
||||||
|
/** runtime descriptor: "nodejs18.x" */
|
||||||
runtime: string;
|
runtime: string;
|
||||||
|
/** date beyond which this version is discontinued: 2023-08-17T19:05:45.951Z */
|
||||||
discontinueDate?: Date;
|
discontinueDate?: Date;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"declaration": true,
|
"declaration": true,
|
||||||
|
"emitDeclarationOnly": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"lib": ["ES2021"],
|
"lib": ["ES2021"],
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
|
|||||||
@@ -1,5 +1,66 @@
|
|||||||
# vercel
|
# vercel
|
||||||
|
|
||||||
|
## 32.2.0
|
||||||
|
|
||||||
|
### Minor Changes
|
||||||
|
|
||||||
|
- show instant preview url on deploy ([#10458](https://github.com/vercel/vercel/pull/10458))
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- N, not n. ([#10460](https://github.com/vercel/vercel/pull/10460))
|
||||||
|
|
||||||
|
- Fix team URL on `vercel help switch` ([#10466](https://github.com/vercel/vercel/pull/10466))
|
||||||
|
|
||||||
|
- Migrates the vc env command to the command data structure for use in the help output. ([#10429](https://github.com/vercel/vercel/pull/10429))
|
||||||
|
|
||||||
|
- Update domains command to new structure ([#10427](https://github.com/vercel/vercel/pull/10427))
|
||||||
|
|
||||||
|
- Updated semver dependency ([#10411](https://github.com/vercel/vercel/pull/10411))
|
||||||
|
|
||||||
|
- migrate `rollback` command structure for help output ([#10426](https://github.com/vercel/vercel/pull/10426))
|
||||||
|
|
||||||
|
- migrate `inti` command structure for help output ([#10428](https://github.com/vercel/vercel/pull/10428))
|
||||||
|
|
||||||
|
- Remove mri workaround ([#10452](https://github.com/vercel/vercel/pull/10452))
|
||||||
|
|
||||||
|
- migrate dev command structure for help output ([#10433](https://github.com/vercel/vercel/pull/10433))
|
||||||
|
|
||||||
|
- Update secrets to more recent structure ([#10461](https://github.com/vercel/vercel/pull/10461))
|
||||||
|
|
||||||
|
- Migrate `vc secrets` to new help command structure ([#10435](https://github.com/vercel/vercel/pull/10435))
|
||||||
|
|
||||||
|
- migrate `promote` command structure for help output ([#10425](https://github.com/vercel/vercel/pull/10425))
|
||||||
|
|
||||||
|
- migrate `git` command structure for help output ([#10431](https://github.com/vercel/vercel/pull/10431))
|
||||||
|
|
||||||
|
- Update project command to new data structure ([#10432](https://github.com/vercel/vercel/pull/10432))
|
||||||
|
|
||||||
|
- migrate teams command ([#10434](https://github.com/vercel/vercel/pull/10434))
|
||||||
|
|
||||||
|
- Updated dependencies [[`5609a1187`](https://github.com/vercel/vercel/commit/5609a1187be9d6cf8d5f16825690c5ea72f17dc5), [`caaba0d68`](https://github.com/vercel/vercel/commit/caaba0d6855eff4350b6a04acc3ea502025bff8f), [`1b4de4a98`](https://github.com/vercel/vercel/commit/1b4de4a986f7a612aac834ebae3ec7bb9e9b8cf8), [`c3c54d6e6`](https://github.com/vercel/vercel/commit/c3c54d6e695ec078777c4b1f4f23acbeee3c3b09), [`6aa0aa4e6`](https://github.com/vercel/vercel/commit/6aa0aa4e65b81903f4fce677a198dcfaebee744b), [`e43191b18`](https://github.com/vercel/vercel/commit/e43191b1866da70a3dab3815a3f2176942240ef3), [`fc1e13c09`](https://github.com/vercel/vercel/commit/fc1e13c09928c654410b373fc1775c2b63c6ef4a)]:
|
||||||
|
- @vercel/build-utils@7.1.1
|
||||||
|
- @vercel/next@4.0.2
|
||||||
|
- @vercel/static-build@2.0.4
|
||||||
|
- @vercel/redwood@2.0.1
|
||||||
|
- @vercel/remix-builder@2.0.3
|
||||||
|
- @vercel/ruby@2.0.1
|
||||||
|
- @vercel/node@3.0.4
|
||||||
|
|
||||||
|
## 32.1.0
|
||||||
|
|
||||||
|
### Minor Changes
|
||||||
|
|
||||||
|
- Improve error messages for JSON parse failures ([#10396](https://github.com/vercel/vercel/pull/10396))
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`9e3827c78`](https://github.com/vercel/vercel/commit/9e3827c785e1bc45f2bed421132167381481770f)]:
|
||||||
|
- @vercel/build-utils@7.1.0
|
||||||
|
- @vercel/node@3.0.3
|
||||||
|
- @vercel/remix-builder@2.0.2
|
||||||
|
- @vercel/static-build@2.0.3
|
||||||
|
|
||||||
## 32.0.2
|
## 32.0.2
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "vercel",
|
"name": "vercel",
|
||||||
"version": "32.0.2",
|
"version": "32.2.0",
|
||||||
"preferGlobal": true,
|
"preferGlobal": true,
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"description": "The command-line interface for Vercel",
|
"description": "The command-line interface for Vercel",
|
||||||
@@ -31,16 +31,16 @@
|
|||||||
"node": ">= 16"
|
"node": ">= 16"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/go": "3.0.0",
|
"@vercel/go": "3.0.0",
|
||||||
"@vercel/hydrogen": "1.0.0",
|
"@vercel/hydrogen": "1.0.0",
|
||||||
"@vercel/next": "4.0.1",
|
"@vercel/next": "4.0.2",
|
||||||
"@vercel/node": "3.0.2",
|
"@vercel/node": "3.0.4",
|
||||||
"@vercel/python": "4.0.0",
|
"@vercel/python": "4.0.0",
|
||||||
"@vercel/redwood": "2.0.0",
|
"@vercel/redwood": "2.0.1",
|
||||||
"@vercel/remix-builder": "2.0.1",
|
"@vercel/remix-builder": "2.0.3",
|
||||||
"@vercel/ruby": "2.0.0",
|
"@vercel/ruby": "2.0.1",
|
||||||
"@vercel/static-build": "2.0.2"
|
"@vercel/static-build": "2.0.4"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@alex_neo/jest-expect-message": "1.0.5",
|
"@alex_neo/jest-expect-message": "1.0.5",
|
||||||
@@ -64,6 +64,7 @@
|
|||||||
"@types/inquirer": "7.3.1",
|
"@types/inquirer": "7.3.1",
|
||||||
"@types/jest": "27.4.1",
|
"@types/jest": "27.4.1",
|
||||||
"@types/jest-expect-message": "1.0.3",
|
"@types/jest-expect-message": "1.0.3",
|
||||||
|
"@types/json-parse-better-errors": "1.0.0",
|
||||||
"@types/load-json-file": "2.0.7",
|
"@types/load-json-file": "2.0.7",
|
||||||
"@types/mime-types": "2.1.0",
|
"@types/mime-types": "2.1.0",
|
||||||
"@types/minimatch": "3.0.3",
|
"@types/minimatch": "3.0.3",
|
||||||
@@ -85,11 +86,11 @@
|
|||||||
"@types/yauzl-promise": "2.1.0",
|
"@types/yauzl-promise": "2.1.0",
|
||||||
"@vercel-internals/constants": "1.0.4",
|
"@vercel-internals/constants": "1.0.4",
|
||||||
"@vercel-internals/get-package-json": "1.0.0",
|
"@vercel-internals/get-package-json": "1.0.0",
|
||||||
"@vercel-internals/types": "1.0.7",
|
"@vercel-internals/types": "1.0.9",
|
||||||
"@vercel/client": "13.0.0",
|
"@vercel/client": "13.0.2",
|
||||||
"@vercel/error-utils": "2.0.1",
|
"@vercel/error-utils": "2.0.1",
|
||||||
"@vercel/frameworks": "2.0.1",
|
"@vercel/frameworks": "2.0.1",
|
||||||
"@vercel/fs-detectors": "5.0.1",
|
"@vercel/fs-detectors": "5.0.2",
|
||||||
"@vercel/fun": "1.0.4",
|
"@vercel/fun": "1.0.4",
|
||||||
"@vercel/ncc": "0.24.0",
|
"@vercel/ncc": "0.24.0",
|
||||||
"@vercel/routing-utils": "3.0.0",
|
"@vercel/routing-utils": "3.0.0",
|
||||||
@@ -135,6 +136,7 @@
|
|||||||
"jaro-winkler": "0.2.8",
|
"jaro-winkler": "0.2.8",
|
||||||
"jest-junit": "16.0.0",
|
"jest-junit": "16.0.0",
|
||||||
"jest-matcher-utils": "29.3.1",
|
"jest-matcher-utils": "29.3.1",
|
||||||
|
"json-parse-better-errors": "1.0.2",
|
||||||
"jsonlines": "0.1.1",
|
"jsonlines": "0.1.1",
|
||||||
"line-async-iterator": "3.0.0",
|
"line-async-iterator": "3.0.0",
|
||||||
"load-json-file": "3.0.0",
|
"load-json-file": "3.0.0",
|
||||||
@@ -154,7 +156,7 @@
|
|||||||
"qr-image": "3.2.0",
|
"qr-image": "3.2.0",
|
||||||
"raw-body": "2.4.1",
|
"raw-body": "2.4.1",
|
||||||
"rimraf": "3.0.2",
|
"rimraf": "3.0.2",
|
||||||
"semver": "5.5.0",
|
"semver": "5.7.2",
|
||||||
"serve-handler": "6.1.1",
|
"serve-handler": "6.1.1",
|
||||||
"strip-ansi": "6.0.1",
|
"strip-ansi": "6.0.1",
|
||||||
"supports-hyperlinks": "3.0.0",
|
"supports-hyperlinks": "3.0.0",
|
||||||
|
|||||||
34
packages/cli/src/commands/dev/command.ts
Normal file
34
packages/cli/src/commands/dev/command.ts
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const devCommand: Command = {
|
||||||
|
name: 'dev',
|
||||||
|
description: `Starts the \`${packageName} dev\` server.`,
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'dir',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'listen',
|
||||||
|
description: 'Specify a URI endpoint on which to listen [0.0.0.0:3000]',
|
||||||
|
argument: 'uri',
|
||||||
|
shorthand: null,
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: `Start the \`${packageName} dev\` server on port 8080`,
|
||||||
|
value: `${packageName} dev --listen 8080`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Make the `vercel dev` server bind to localhost on port 5000',
|
||||||
|
value: `${packageName} dev --listen 127.0.0.1:5000 `,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -12,43 +12,16 @@ import highlight from '../../util/output/highlight';
|
|||||||
import dev from './dev';
|
import dev from './dev';
|
||||||
import readConfig from '../../util/config/read-config';
|
import readConfig from '../../util/config/read-config';
|
||||||
import readJSONFile from '../../util/read-json-file';
|
import readJSONFile from '../../util/read-json-file';
|
||||||
import { packageName, getCommandName, logo } from '../../util/pkg-name';
|
import { packageName, getCommandName } from '../../util/pkg-name';
|
||||||
import { CantParseJSONFile } from '../../util/errors-ts';
|
import { CantParseJSONFile } from '../../util/errors-ts';
|
||||||
import { isErrnoException } from '@vercel/error-utils';
|
import { isErrnoException } from '@vercel/error-utils';
|
||||||
|
import { help } from '../help';
|
||||||
|
import { devCommand } from './command';
|
||||||
|
|
||||||
const COMMAND_CONFIG = {
|
const COMMAND_CONFIG = {
|
||||||
dev: ['dev'],
|
dev: ['dev'],
|
||||||
};
|
};
|
||||||
|
|
||||||
const help = () => {
|
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} dev`)} [options] <dir>
|
|
||||||
|
|
||||||
Starts the \`${packageName} dev\` server.
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-l, --listen [uri] Specify a URI endpoint on which to listen [0.0.0.0:3000]
|
|
||||||
-t, --token [token] Specify an Authorization Token
|
|
||||||
-y, --yes Skip questions when setting up new project using default scope and settings
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Start the \`${packageName} dev\` server on port 8080
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} dev --listen 8080`)}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Make the \`vercel dev\` server bind to localhost on port 5000
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} dev --listen 127.0.0.1:5000`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default async function main(client: Client) {
|
export default async function main(client: Client) {
|
||||||
if (process.env.__VERCEL_DEV_RUNNING) {
|
if (process.env.__VERCEL_DEV_RUNNING) {
|
||||||
client.output.error(
|
client.output.error(
|
||||||
@@ -100,7 +73,7 @@ export default async function main(client: Client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
client.output.print(help(devCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -118,7 +91,7 @@ export default async function main(client: Client) {
|
|||||||
const pkg = await readJSONFile<PackageJson>(path.join(dir, 'package.json'));
|
const pkg = await readJSONFile<PackageJson>(path.join(dir, 'package.json'));
|
||||||
|
|
||||||
if (pkg instanceof CantParseJSONFile) {
|
if (pkg instanceof CantParseJSONFile) {
|
||||||
client.output.error('Could not parse package.json');
|
client.output.error(pkg.message);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
153
packages/cli/src/commands/domains/command.ts
Normal file
153
packages/cli/src/commands/domains/command.ts
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const domainsCommand: Command = {
|
||||||
|
name: 'domains',
|
||||||
|
description: 'Manage domains',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'command',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'ls',
|
||||||
|
description: 'Show all domains in a list',
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'inspect',
|
||||||
|
description: 'Displays information related to a domain',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'add',
|
||||||
|
description: 'Add a new domain that you already own',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'project',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'rm',
|
||||||
|
description: 'Remove a domain',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'buy',
|
||||||
|
description: "Buy a domain that you don't yet own",
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'move',
|
||||||
|
description: 'Move a domain to another user or team',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'destination',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'transfer-in',
|
||||||
|
description: 'Transfer in a domain to Vercel',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'next',
|
||||||
|
description: 'Show next page of results',
|
||||||
|
shorthand: 'N',
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'yes',
|
||||||
|
description: 'Skip the confirmation prompt when removing a domain',
|
||||||
|
shorthand: 'y',
|
||||||
|
type: 'boolean',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'limit',
|
||||||
|
shorthand: 'n',
|
||||||
|
description:
|
||||||
|
'Number of results to return per page (default: 20, max: 100)',
|
||||||
|
argument: 'NUMBER',
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'force',
|
||||||
|
shorthand: 'f',
|
||||||
|
type: 'boolean',
|
||||||
|
deprecated: false,
|
||||||
|
description:
|
||||||
|
'Force a domain on a project and remove it from an existing one',
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Add a domain that you already own',
|
||||||
|
value: [
|
||||||
|
`${packageName} domains add domain-name.com`,
|
||||||
|
"Make sure the domain's DNS nameservers are at least 2 of the ones listed on https://vercel.com/edge-network",
|
||||||
|
`NOTE: Running ${packageName} alias will automatically register your domain if it's configured with these nameservers (no need to 'domains add')`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Paginate results, where `1584722256178` is the time in milliseconds since the UNIX epoch',
|
||||||
|
value: `${packageName} domains ls--next 1584722256178`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,5 +1,3 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
|
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import getSubcommand from '../../util/get-subcommand';
|
import getSubcommand from '../../util/get-subcommand';
|
||||||
@@ -11,69 +9,8 @@ import inspect from './inspect';
|
|||||||
import ls from './ls';
|
import ls from './ls';
|
||||||
import rm from './rm';
|
import rm from './rm';
|
||||||
import move from './move';
|
import move from './move';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
import { domainsCommand } from './command';
|
||||||
|
import { help } from '../help';
|
||||||
const help = () => {
|
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} domains`)} [options] <command>
|
|
||||||
|
|
||||||
${chalk.dim('Commands:')}
|
|
||||||
|
|
||||||
ls Show all domains in a list
|
|
||||||
inspect [name] Displays information related to a domain
|
|
||||||
add [name] [project] Add a new domain that you already own
|
|
||||||
rm [name] Remove a domain
|
|
||||||
buy [name] Buy a domain that you don't yet own
|
|
||||||
move [name] [destination] Move a domain to another user or team.
|
|
||||||
transfer-in [name] Transfer in a domain to Vercel
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-f, --force Force a domain on a project and remove it from an existing one
|
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
|
||||||
'FILE'
|
|
||||||
)} Path to the local ${'`vercel.json`'} file
|
|
||||||
-Q ${chalk.bold.underline('DIR')}, --global-config=${chalk.bold.underline(
|
|
||||||
'DIR'
|
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
-S, --scope Set a custom scope
|
|
||||||
-N, --next Show next page of results
|
|
||||||
--limit=${chalk.bold.underline(
|
|
||||||
'VALUE'
|
|
||||||
)} Number of results to return per page (default: 20, max: 100)
|
|
||||||
-y, --yes Skip the confirmation prompt when removing a domain
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Add a domain that you already own
|
|
||||||
|
|
||||||
${chalk.cyan(
|
|
||||||
`$ ${packageName} domains add ${chalk.underline('domain-name.com')}`
|
|
||||||
)}
|
|
||||||
|
|
||||||
Make sure the domain's DNS nameservers are at least 2 of the
|
|
||||||
ones listed on ${chalk.underline('https://vercel.com/edge-network')}.
|
|
||||||
|
|
||||||
${chalk.yellow('NOTE:')} Running ${chalk.dim(
|
|
||||||
`${packageName} alias`
|
|
||||||
)} will automatically register your domain
|
|
||||||
if it's configured with these nameservers (no need to ${chalk.dim(
|
|
||||||
'`domain add`'
|
|
||||||
)}).
|
|
||||||
|
|
||||||
${chalk.gray('–')} Paginate results, where ${chalk.dim(
|
|
||||||
'`1584722256178`'
|
|
||||||
)} is the time in milliseconds since the UNIX epoch.
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} domains ls --next 1584722256178`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
const COMMAND_CONFIG = {
|
const COMMAND_CONFIG = {
|
||||||
add: ['add'],
|
add: ['add'],
|
||||||
@@ -104,7 +41,9 @@ export default async function main(client: Client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
client.output.print(
|
||||||
|
help(domainsCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
157
packages/cli/src/commands/env/command.ts
vendored
Normal file
157
packages/cli/src/commands/env/command.ts
vendored
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
import { getEnvTargetPlaceholder } from '../../util/env/env-target';
|
||||||
|
|
||||||
|
const targetPlaceholder = getEnvTargetPlaceholder();
|
||||||
|
|
||||||
|
export const envCommand: Command = {
|
||||||
|
name: 'env',
|
||||||
|
description: 'Interact with environment variables.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'command',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'ls',
|
||||||
|
description: 'List all variables for the specified Environment',
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'add',
|
||||||
|
description: 'Add an Environment Variable (see examples below)',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'environment',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'rm',
|
||||||
|
description: 'Remove an Environment Variable (see examples below)',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'environment',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'pull',
|
||||||
|
description:
|
||||||
|
'Pull all Development Environment Variables from the cloud and write to a file [.env.local]',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'filename',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'environment',
|
||||||
|
description:
|
||||||
|
'Set the Environment (development, preview, production) when pulling Environment Variables',
|
||||||
|
shorthand: null,
|
||||||
|
type: 'boolean',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'git-branch',
|
||||||
|
description:
|
||||||
|
'Specify the Git branch to pull specific Environment Variables for',
|
||||||
|
shorthand: null,
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'yes',
|
||||||
|
description: 'Skip the confirmation prompt when removing an alias',
|
||||||
|
shorthand: 'y',
|
||||||
|
type: 'boolean',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Pull all Development Environment Variables down from the cloud',
|
||||||
|
value: [
|
||||||
|
`${packageName} env pull <file>`,
|
||||||
|
`${packageName} env pull .env.development.local`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Add a new variable to multiple Environments',
|
||||||
|
value: [
|
||||||
|
`${packageName} env add <name>`,
|
||||||
|
`${packageName} env add API_TOKEN`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Add a new variable for a specific Environment',
|
||||||
|
value: [
|
||||||
|
`${packageName} env add <name> ${targetPlaceholder}`,
|
||||||
|
`${packageName} env add DB_PASS production`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Add a new variable for a specific Environment and Git Branch',
|
||||||
|
value: [
|
||||||
|
`${packageName} env add <name> ${targetPlaceholder} <gitbranch>`,
|
||||||
|
`${packageName} env add DB_PASS preview feat1`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Add a new Environment Variable from stdin',
|
||||||
|
value: [
|
||||||
|
`cat <file> | ${packageName} env add <name> ${targetPlaceholder}`,
|
||||||
|
`cat ~/.npmrc | ${packageName} env add NPM_RC preview`,
|
||||||
|
`${packageName} env add API_URL production < url.txt`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Remove a variable from multiple Environments',
|
||||||
|
value: [
|
||||||
|
`${packageName} env rm <name>`,
|
||||||
|
`${packageName} env rm API_TOKEN`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Remove a variable from a specific Environment',
|
||||||
|
value: [
|
||||||
|
`${packageName} env rm <name> ${targetPlaceholder}`,
|
||||||
|
`${packageName} env rm NPM_RC preview`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Remove a variable from a specific Environment and Git Branch',
|
||||||
|
value: [
|
||||||
|
`${packageName} env rm <name> ${targetPlaceholder} <gitbranch>`,
|
||||||
|
`${packageName} env rm NPM_RC preview feat1`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
98
packages/cli/src/commands/env/index.ts
vendored
98
packages/cli/src/commands/env/index.ts
vendored
@@ -8,99 +8,15 @@ import getArgs from '../../util/get-args';
|
|||||||
import getInvalidSubcommand from '../../util/get-invalid-subcommand';
|
import getInvalidSubcommand from '../../util/get-invalid-subcommand';
|
||||||
import getSubcommand from '../../util/get-subcommand';
|
import getSubcommand from '../../util/get-subcommand';
|
||||||
import handleError from '../../util/handle-error';
|
import handleError from '../../util/handle-error';
|
||||||
import { getCommandName, packageName, logo } from '../../util/pkg-name';
|
import { help } from '../help';
|
||||||
|
import { getCommandName } from '../../util/pkg-name';
|
||||||
import { getLinkedProject } from '../../util/projects/link';
|
import { getLinkedProject } from '../../util/projects/link';
|
||||||
|
|
||||||
import add from './add';
|
import add from './add';
|
||||||
import ls from './ls';
|
import ls from './ls';
|
||||||
import pull from './pull';
|
import pull from './pull';
|
||||||
import rm from './rm';
|
import rm from './rm';
|
||||||
|
import { envCommand } from './command';
|
||||||
const help = () => {
|
|
||||||
const targetPlaceholder = getEnvTargetPlaceholder();
|
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} env`)} [options] <command>
|
|
||||||
|
|
||||||
${chalk.dim('Commands:')}
|
|
||||||
|
|
||||||
ls [environment] [gitbranch] List all variables for the specified Environment
|
|
||||||
add [name] [environment] [gitbranch] Add an Environment Variable (see examples below)
|
|
||||||
rm [name] [environment] [gitbranch] Remove an Environment Variable (see examples below)
|
|
||||||
pull [filename] Pull all Development Environment Variables from the cloud and write to a file [.env.local]
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
--environment Set the Environment (development, preview, production) when pulling Environment Variables
|
|
||||||
--git-branch Specify the Git branch to pull specific Environment Variables for
|
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
|
||||||
'FILE'
|
|
||||||
)} Path to the local ${'`vercel.json`'} file
|
|
||||||
-Q ${chalk.bold.underline('DIR')}, --global-config=${chalk.bold.underline(
|
|
||||||
'DIR'
|
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
-y, --yes Skip the confirmation prompt when overwriting env file on pull or removing an env variable
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Pull all Development Environment Variables down from the cloud
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} env pull <file>`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env pull .env.development.local`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Add a new variable to multiple Environments
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} env add <name>`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env add API_TOKEN`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Add a new variable for a specific Environment
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} env add <name> ${targetPlaceholder}`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env add DB_PASS production`)}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Add a new variable for a specific Environment and Git Branch
|
|
||||||
|
|
||||||
${chalk.cyan(
|
|
||||||
`$ ${packageName} env add <name> ${targetPlaceholder} <gitbranch>`
|
|
||||||
)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env add DB_PASS preview feat1`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Add a new Environment Variable from stdin
|
|
||||||
|
|
||||||
${chalk.cyan(
|
|
||||||
`$ cat <file> | ${packageName} env add <name> ${targetPlaceholder}`
|
|
||||||
)}
|
|
||||||
${chalk.cyan(`$ cat ~/.npmrc | ${packageName} env add NPM_RC preview`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env add API_URL production < url.txt`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Remove a variable from multiple Environments
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} env rm <name>`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env rm API_TOKEN`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Remove a variable from a specific Environment
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} env rm <name> ${targetPlaceholder}`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env rm NPM_RC preview`)}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Remove a variable from a specific Environment and Git Branch
|
|
||||||
|
|
||||||
${chalk.cyan(
|
|
||||||
`$ ${packageName} env rm <name> ${targetPlaceholder} <gitbranch>`
|
|
||||||
)}
|
|
||||||
${chalk.cyan(`$ ${packageName} env rm NPM_RC preview feat1`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
const COMMAND_CONFIG = {
|
const COMMAND_CONFIG = {
|
||||||
ls: ['ls', 'list'],
|
ls: ['ls', 'list'],
|
||||||
@@ -125,7 +41,7 @@ export default async function main(client: Client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
client.output.print(help(envCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -177,7 +93,9 @@ export default async function main(client: Client) {
|
|||||||
);
|
);
|
||||||
default:
|
default:
|
||||||
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
||||||
help();
|
client.output.print(
|
||||||
|
help(envCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
3
packages/cli/src/commands/env/pull.ts
vendored
3
packages/cli/src/commands/env/pull.ts
vendored
@@ -24,6 +24,7 @@ import {
|
|||||||
} from '../../util/env/diff-env-files';
|
} from '../../util/env/diff-env-files';
|
||||||
import { isErrnoException } from '@vercel/error-utils';
|
import { isErrnoException } from '@vercel/error-utils';
|
||||||
import { addToGitIgnore } from '../../util/link/add-to-gitignore';
|
import { addToGitIgnore } from '../../util/link/add-to-gitignore';
|
||||||
|
import JSONparse from 'json-parse-better-errors';
|
||||||
|
|
||||||
const CONTENTS_PREFIX = '# Created by Vercel CLI\n';
|
const CONTENTS_PREFIX = '# Created by Vercel CLI\n';
|
||||||
|
|
||||||
@@ -121,7 +122,7 @@ export default async function pull(
|
|||||||
// We need this because double quotes are stripped from the local .env file,
|
// We need this because double quotes are stripped from the local .env file,
|
||||||
// but `records` is already in the form of a JSON object that doesn't filter
|
// but `records` is already in the form of a JSON object that doesn't filter
|
||||||
// double quotes.
|
// double quotes.
|
||||||
const newEnv = JSON.parse(JSON.stringify(records).replace(/\\"/g, ''));
|
const newEnv = JSONparse(JSON.stringify(records).replace(/\\"/g, ''));
|
||||||
deltaString = buildDeltaString(oldEnv, newEnv);
|
deltaString = buildDeltaString(oldEnv, newEnv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
50
packages/cli/src/commands/git/command.ts
Normal file
50
packages/cli/src/commands/git/command.ts
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const gitCommand: Command = {
|
||||||
|
name: 'git',
|
||||||
|
description: 'Manage your Git provider connections.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'command',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'connect',
|
||||||
|
description:
|
||||||
|
'Connect your Vercel Project to your Git repository or provide the remote URL to your Git repository',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'git url',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'disconnect',
|
||||||
|
description: 'Disconnect the Git provider repository from your project',
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Connect your Vercel Project to your Git repository defined in your local .git config',
|
||||||
|
value: `${packageName} git connect`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Connect your Vercel Project to a Git repository using the remote URL',
|
||||||
|
value: `${packageName} git connect https://github.com/user/repo.git`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Disconnect the Git provider repository',
|
||||||
|
value: `${packageName} git disconnect`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,51 +1,12 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
import { ensureLink } from '../../util/link/ensure-link';
|
import { ensureLink } from '../../util/link/ensure-link';
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import getInvalidSubcommand from '../../util/get-invalid-subcommand';
|
import getInvalidSubcommand from '../../util/get-invalid-subcommand';
|
||||||
import handleError from '../../util/handle-error';
|
import handleError from '../../util/handle-error';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
|
||||||
import connect from './connect';
|
import connect from './connect';
|
||||||
import disconnect from './disconnect';
|
import disconnect from './disconnect';
|
||||||
|
import { help } from '../help';
|
||||||
const help = () => {
|
import { gitCommand } from './command';
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} git`)} <command>
|
|
||||||
|
|
||||||
${chalk.dim('Commands:')}
|
|
||||||
|
|
||||||
connect [url] Connect your Vercel Project to your Git repository or provide the remote URL to your Git repository
|
|
||||||
disconnect Disconnect the Git provider repository from your project
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
-y, --yes Skip confirmation when connecting a Git provider
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Connect your Vercel Project to your Git repository defined in your local .git config
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} git connect`)}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Connect your Vercel Project to a Git repository using the remote URL
|
|
||||||
|
|
||||||
${chalk.cyan(
|
|
||||||
`$ ${packageName} git connect https://github.com/user/repo.git`
|
|
||||||
)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Disconnect the Git provider repository
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} git disconnect`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
const COMMAND_CONFIG = {
|
const COMMAND_CONFIG = {
|
||||||
connect: ['connect'],
|
connect: ['connect'],
|
||||||
@@ -71,7 +32,7 @@ export default async function main(client: Client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
client.output.print(help(gitCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,7 +57,7 @@ export default async function main(client: Client) {
|
|||||||
return await disconnect(client, args, project, org);
|
return await disconnect(client, args, project, org);
|
||||||
default:
|
default:
|
||||||
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
||||||
help();
|
client.output.print(help(gitCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ export interface CommandOption {
|
|||||||
argument?: string;
|
argument?: string;
|
||||||
deprecated: boolean;
|
deprecated: boolean;
|
||||||
description?: string;
|
description?: string;
|
||||||
|
/** supports multiple entries */
|
||||||
multi: boolean;
|
multi: boolean;
|
||||||
}
|
}
|
||||||
export interface CommandArgument {
|
export interface CommandArgument {
|
||||||
@@ -322,7 +323,10 @@ export function buildSubcommandLines(
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
const finalColumnWidth = options.columns - maxWidthOfUnwrappedColumns;
|
// Really long descriptions go RIGHT up to the edge, which looks unpleasant.
|
||||||
|
const rightMargin = INDENT.repeat(4).length;
|
||||||
|
const finalColumnWidth =
|
||||||
|
options.columns - maxWidthOfUnwrappedColumns - rightMargin;
|
||||||
|
|
||||||
const table = new Table(
|
const table = new Table(
|
||||||
Object.assign({}, tableOptions, {
|
Object.assign({}, tableOptions, {
|
||||||
|
|||||||
47
packages/cli/src/commands/init/command.ts
Normal file
47
packages/cli/src/commands/init/command.ts
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const initCommand: Command = {
|
||||||
|
name: 'init',
|
||||||
|
description: 'Initialize example Vercel Projects',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'example',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'dir',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: undefined,
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'force',
|
||||||
|
description: 'Overwrite destination directory if exists [off]',
|
||||||
|
argument: undefined,
|
||||||
|
shorthand: 'f',
|
||||||
|
type: 'boolean',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Choose from all available examples',
|
||||||
|
value: `${packageName} init`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Initialize example project into a new directory',
|
||||||
|
value: `${packageName} init <example>`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Initialize example project into specified directory',
|
||||||
|
value: `${packageName} <example> <dir>`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Initialize example project without checking',
|
||||||
|
value: `${packageName} init <example> --force`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,48 +1,16 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
|
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import getSubcommand from '../../util/get-subcommand';
|
import getSubcommand from '../../util/get-subcommand';
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
import handleError from '../../util/handle-error';
|
import handleError from '../../util/handle-error';
|
||||||
import init from './init';
|
import init from './init';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
|
||||||
import { isError } from '@vercel/error-utils';
|
import { isError } from '@vercel/error-utils';
|
||||||
|
import { help } from '../help';
|
||||||
|
import { initCommand } from './command';
|
||||||
|
|
||||||
const COMMAND_CONFIG = {
|
const COMMAND_CONFIG = {
|
||||||
init: ['init'],
|
init: ['init'],
|
||||||
};
|
};
|
||||||
|
|
||||||
const help = () => {
|
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} init`)} [example] [dir] [-f | --force]
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-f, --force Overwrite destination directory if exists [off]
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Choose from all available examples
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} init`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Initialize example project into a new directory
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} init <example>`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Initialize example project into specified directory
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} init <example> <dir>`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Initialize example project without checking
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} init <example> --force`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default async function main(client: Client) {
|
export default async function main(client: Client) {
|
||||||
const { output } = client;
|
const { output } = client;
|
||||||
let argv;
|
let argv;
|
||||||
@@ -60,7 +28,7 @@ export default async function main(client: Client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
client.output.print(help(initCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
67
packages/cli/src/commands/project/command.ts
Normal file
67
packages/cli/src/commands/project/command.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const projectCommand: Command = {
|
||||||
|
name: 'project',
|
||||||
|
description: 'Manage your Vercel Projects.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'command',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'ls',
|
||||||
|
description: 'Show all projects in the selected team/user',
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'add',
|
||||||
|
description: 'Add a new project',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'rm',
|
||||||
|
description: 'Remove a project',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'next',
|
||||||
|
description: 'Show next page of results',
|
||||||
|
argument: 'MS',
|
||||||
|
shorthand: 'N',
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Add a new project',
|
||||||
|
value: `${packageName} project add my-project`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Paginate projects, where `1584722256178` is the time in milliseconds since the UNIX epoch',
|
||||||
|
value: `${packageName} project ls --next 1584722256178`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,46 +1,13 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import getInvalidSubcommand from '../../util/get-invalid-subcommand';
|
import getInvalidSubcommand from '../../util/get-invalid-subcommand';
|
||||||
import getScope from '../../util/get-scope';
|
import getScope from '../../util/get-scope';
|
||||||
import handleError from '../../util/handle-error';
|
import handleError from '../../util/handle-error';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
import { help } from '../help';
|
||||||
import add from './add';
|
import add from './add';
|
||||||
import list from './list';
|
import list from './list';
|
||||||
import rm from './rm';
|
import rm from './rm';
|
||||||
|
import { projectCommand } from './command';
|
||||||
const help = () => {
|
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} project`)} [options] <command>
|
|
||||||
|
|
||||||
${chalk.dim('Commands:')}
|
|
||||||
|
|
||||||
ls Show all projects in the selected team/user
|
|
||||||
add [name] Add a new project
|
|
||||||
rm [name] Remove a project
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
-S, --scope Set a custom scope
|
|
||||||
-N, --next Show next page of results
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Add a new project
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} project add my-project`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Paginate projects, where ${chalk.dim(
|
|
||||||
'`1584722256178`'
|
|
||||||
)} is the time in milliseconds since the UNIX epoch.
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} project ls --next 1584722256178`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
const COMMAND_CONFIG = {
|
const COMMAND_CONFIG = {
|
||||||
ls: ['ls', 'list'],
|
ls: ['ls', 'list'],
|
||||||
@@ -63,7 +30,9 @@ export default async function main(client: Client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
client.output.print(
|
||||||
|
help(projectCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -84,7 +53,9 @@ export default async function main(client: Client) {
|
|||||||
return await rm(client, args);
|
return await rm(client, args);
|
||||||
default:
|
default:
|
||||||
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
||||||
help();
|
client.output.print(
|
||||||
|
help(projectCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
53
packages/cli/src/commands/promote/command.ts
Normal file
53
packages/cli/src/commands/promote/command.ts
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const promoteCommand: Command = {
|
||||||
|
name: 'promote',
|
||||||
|
description: 'Promote an existing deployment to current.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'deployment id/url',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'status',
|
||||||
|
description: 'Show the status of any current pending promotions',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'project',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'timeout',
|
||||||
|
description: 'Time to wait for promotion completion [3m]',
|
||||||
|
argument: 'timeout',
|
||||||
|
shorthand: null,
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Show the status of any current pending promotions',
|
||||||
|
value: [
|
||||||
|
`${packageName} promote`,
|
||||||
|
`${packageName} promote status`,
|
||||||
|
`${packageName} promote status <project>`,
|
||||||
|
`${packageName} promote status --timeout 30s`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Promote a deployment using id or url',
|
||||||
|
value: `${packageName} promote <deployment id/url>`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,53 +1,13 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
import type Client from '../../util/client';
|
import type Client from '../../util/client';
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import getProjectByCwdOrLink from '../../util/projects/get-project-by-cwd-or-link';
|
import getProjectByCwdOrLink from '../../util/projects/get-project-by-cwd-or-link';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
|
||||||
import handleError from '../../util/handle-error';
|
import handleError from '../../util/handle-error';
|
||||||
import { isErrnoException } from '@vercel/error-utils';
|
import { isErrnoException } from '@vercel/error-utils';
|
||||||
import ms from 'ms';
|
import ms from 'ms';
|
||||||
import requestPromote from './request-promote';
|
import requestPromote from './request-promote';
|
||||||
import promoteStatus from './status';
|
import promoteStatus from './status';
|
||||||
|
import { promoteCommand } from './command';
|
||||||
const help = () => {
|
import { help } from '../help';
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} promote`)} [deployment id/url]
|
|
||||||
|
|
||||||
Promote an existing deployment to current.
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
|
||||||
'FILE'
|
|
||||||
)} Path to the local ${'`vercel.json`'} file
|
|
||||||
-Q ${chalk.bold.underline('DIR')}, --global-config=${chalk.bold.underline(
|
|
||||||
'DIR'
|
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
--timeout=${chalk.bold.underline(
|
|
||||||
'TIME'
|
|
||||||
)} Time to wait for promotion completion [3m]
|
|
||||||
-y, --yes Skip questions when setting up new project using default scope and settings
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Show the status of any current pending promotions
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} promote`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} promote status`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} promote status <project>`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} promote status --timeout 30s`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Promote a deployment using id or url
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} promote <deployment id/url>`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* `vc promote` command
|
* `vc promote` command
|
||||||
@@ -68,7 +28,9 @@ export default async (client: Client): Promise<number> => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help'] || argv._[0] === 'help') {
|
if (argv['--help'] || argv._[0] === 'help') {
|
||||||
help();
|
client.output.print(
|
||||||
|
help(promoteCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -76,12 +76,23 @@ export default async function redeploy(client: Client): Promise<number> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
output.stopSpinner();
|
output.stopSpinner();
|
||||||
|
|
||||||
|
const isProdDeployment = deployment.target === 'production';
|
||||||
|
const previewUrl = `https://${deployment.url}`;
|
||||||
output.print(
|
output.print(
|
||||||
`${prependEmoji(
|
`${prependEmoji(
|
||||||
`Inspect: ${chalk.bold(deployment.inspectorUrl)} ${deployStamp()}`,
|
`Inspect: ${chalk.bold(deployment.inspectorUrl)} ${deployStamp()}`,
|
||||||
emoji('inspect')
|
emoji('inspect')
|
||||||
)}\n`
|
)}\n`
|
||||||
);
|
);
|
||||||
|
output.print(
|
||||||
|
prependEmoji(
|
||||||
|
`${isProdDeployment ? 'Production' : 'Preview'}: ${chalk.bold(
|
||||||
|
previewUrl
|
||||||
|
)} ${deployStamp()}`,
|
||||||
|
emoji('success')
|
||||||
|
) + `\n`
|
||||||
|
);
|
||||||
|
|
||||||
if (!client.stdout.isTTY) {
|
if (!client.stdout.isTTY) {
|
||||||
client.stdout.write(`https://${deployment.url}`);
|
client.stdout.write(`https://${deployment.url}`);
|
||||||
|
|||||||
53
packages/cli/src/commands/rollback/command.ts
Normal file
53
packages/cli/src/commands/rollback/command.ts
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const rollbackCommand: Command = {
|
||||||
|
name: 'rollback',
|
||||||
|
description: 'Quickly revert back to a previous deployment.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'deployment id/url',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'status',
|
||||||
|
description: 'Show the status of any current pending rollbacks',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'project',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'timeout',
|
||||||
|
description: 'Time to wait for rollback completion [3m]',
|
||||||
|
argument: 'timeout',
|
||||||
|
shorthand: null,
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Show the status of any current pending rollbacks',
|
||||||
|
value: [
|
||||||
|
`${packageName} rollback`,
|
||||||
|
`${packageName} rollback status`,
|
||||||
|
`${packageName} rollback status <project>`,
|
||||||
|
`${packageName} rollback status --timeout 30s`,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Rollback a deployment using id or url',
|
||||||
|
value: `${packageName} rollback <deployment id/url>`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,53 +1,13 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
import type Client from '../../util/client';
|
import type Client from '../../util/client';
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import getProjectByCwdOrLink from '../../util/projects/get-project-by-cwd-or-link';
|
import getProjectByCwdOrLink from '../../util/projects/get-project-by-cwd-or-link';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
|
||||||
import handleError from '../../util/handle-error';
|
import handleError from '../../util/handle-error';
|
||||||
import { isErrnoException } from '@vercel/error-utils';
|
import { isErrnoException } from '@vercel/error-utils';
|
||||||
import ms from 'ms';
|
import ms from 'ms';
|
||||||
import requestRollback from './request-rollback';
|
import requestRollback from './request-rollback';
|
||||||
import rollbackStatus from './status';
|
import rollbackStatus from './status';
|
||||||
|
import { help } from '../help';
|
||||||
const help = () => {
|
import { rollbackCommand } from './command';
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} rollback`)} [deployment id/url]
|
|
||||||
|
|
||||||
Quickly revert back to a previous deployment.
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
|
||||||
'FILE'
|
|
||||||
)} Path to the local ${'`vercel.json`'} file
|
|
||||||
-Q ${chalk.bold.underline('DIR')}, --global-config=${chalk.bold.underline(
|
|
||||||
'DIR'
|
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
--timeout=${chalk.bold.underline(
|
|
||||||
'TIME'
|
|
||||||
)} Time to wait for rollback completion [3m]
|
|
||||||
-y, --yes Skip questions when setting up new project using default scope and settings
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Show the status of any current pending rollbacks
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} rollback`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} rollback status`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} rollback status <project>`)}
|
|
||||||
${chalk.cyan(`$ ${packageName} rollback status --timeout 30s`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Rollback a deployment using id or url
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} rollback <deployment id/url>`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* `vc rollback` command
|
* `vc rollback` command
|
||||||
@@ -68,7 +28,9 @@ export default async (client: Client): Promise<number> => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help'] || argv._[0] === 'help') {
|
if (argv['--help'] || argv._[0] === 'help') {
|
||||||
help();
|
client.output.print(
|
||||||
|
help(rollbackCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
96
packages/cli/src/commands/secrets/command.js
Normal file
96
packages/cli/src/commands/secrets/command.js
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import { packageName, getCommandName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const secretsCommand = {
|
||||||
|
name: 'secrets',
|
||||||
|
description: `NOTE: The ${getCommandName(
|
||||||
|
'env'
|
||||||
|
)} command is recommended instead of ${getCommandName('secrets')}`,
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'command',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'ls',
|
||||||
|
description: 'Show all secrets in a list',
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'add',
|
||||||
|
description: 'Add a new secret',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'value',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'rename',
|
||||||
|
description: 'Change the name of a secret',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'old-name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'new-name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'rm',
|
||||||
|
description: 'Remove a secret',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'next',
|
||||||
|
description: 'Show next page of results',
|
||||||
|
argument: 'MS',
|
||||||
|
shorthand: 'n',
|
||||||
|
type: 'string',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: 'Add a new secret',
|
||||||
|
value: `${packageName} secrets add my-secret "my value"
|
||||||
|
|
||||||
|
- Once added, a secret's value can't be retrieved in plain text anymore
|
||||||
|
- If the secret's value is more than one word, wrap it in quotes
|
||||||
|
- When in doubt, always wrap your value in quotes`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Expose a secret as an environment variable (notice the `@` symbol)',
|
||||||
|
value: `${packageName} -e MY_SECRET=@my-secret`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Paginate results, where 1584722256178 is the time in milliseconds since the UNIX epoch',
|
||||||
|
value: `$ ${packageName} secrets ls --next 1584722256178`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,78 +1,17 @@
|
|||||||
|
import isErrnoException from '@vercel/error-utils';
|
||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
import table from 'text-table';
|
import table from 'text-table';
|
||||||
import ms from 'ms';
|
import ms from 'ms';
|
||||||
import strlen from '../util/strlen.ts';
|
import strlen from '../../util/strlen';
|
||||||
import { handleError, error } from '../util/error';
|
import { handleError, error } from '../../util/error';
|
||||||
import NowSecrets from '../util/secrets';
|
import NowSecrets from '../../util/secrets';
|
||||||
import exit from '../util/exit';
|
import getScope from '../../util/get-scope';
|
||||||
import getScope from '../util/get-scope.ts';
|
import confirm from '../../util/input/confirm';
|
||||||
import confirm from '../util/input/confirm';
|
import getCommandFlags from '../../util/get-command-flags';
|
||||||
import getCommandFlags from '../util/get-command-flags';
|
import { getCommandName } from '../../util/pkg-name';
|
||||||
import { packageName, getCommandName, logo } from '../util/pkg-name.ts';
|
import getArgs from '../../util/get-args';
|
||||||
import getArgs from '../util/get-args';
|
import { help } from '../help';
|
||||||
|
import { secretsCommand } from './command';
|
||||||
const help = () => {
|
|
||||||
console.log(`
|
|
||||||
${chalk.yellow(
|
|
||||||
`${chalk.bold('NOTE:')} The ${getCommandName(
|
|
||||||
'env'
|
|
||||||
)} command is recommended instead of ${getCommandName('secrets')}`
|
|
||||||
)}
|
|
||||||
|
|
||||||
${chalk.bold(`${logo} ${packageName} secrets`)} [options] <command>
|
|
||||||
|
|
||||||
${chalk.dim('Commands:')}
|
|
||||||
|
|
||||||
ls Show all secrets in a list
|
|
||||||
add [name] [value] Add a new secret
|
|
||||||
rename [old-name] [new-name] Change the name of a secret
|
|
||||||
rm [name] Remove a secret
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
|
||||||
'FILE'
|
|
||||||
)} Path to the local ${'`vercel.json`'} file
|
|
||||||
-Q ${chalk.bold.underline('DIR')}, --global-config=${chalk.bold.underline(
|
|
||||||
'DIR'
|
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline(
|
|
||||||
'TOKEN'
|
|
||||||
)} Login token
|
|
||||||
-S, --scope Set a custom scope
|
|
||||||
-N, --next Show next page of results
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Add a new secret
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} secrets add my-secret "my value"`)}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Once added, a secret's value can't be retrieved in plain text anymore
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} If the secret's value is more than one word, wrap it in quotes
|
|
||||||
${chalk.gray('–')} When in doubt, always wrap your value in quotes
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} Expose a secret as an environment variable (notice the ${chalk.cyan.bold(
|
|
||||||
'`@`'
|
|
||||||
)} symbol)
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} -e MY_SECRET=${chalk.bold('@my-secret')}`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Paginate results, where ${chalk.dim(
|
|
||||||
'`1584722256178`'
|
|
||||||
)} is the time in milliseconds since the UNIX epoch
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} secrets ls --next 1584722256178`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Options
|
// Options
|
||||||
let argv;
|
let argv;
|
||||||
@@ -94,8 +33,10 @@ const main = async client => {
|
|||||||
nextTimestamp = argv.next;
|
nextTimestamp = argv.next;
|
||||||
|
|
||||||
if (argv.help || !subcommand) {
|
if (argv.help || !subcommand) {
|
||||||
help();
|
client.output.print(
|
||||||
await exit(2);
|
help(secretsCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
const {
|
const {
|
||||||
@@ -107,9 +48,11 @@ const main = async client => {
|
|||||||
try {
|
try {
|
||||||
({ contextName } = await getScope(client));
|
({ contextName } = await getScope(client));
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (err.code === 'NOT_AUTHORIZED' || err.code === 'TEAM_DELETED') {
|
if (isErrnoException(err)) {
|
||||||
output.error(err.message);
|
if (err.code === 'NOT_AUTHORIZED' || err.code === 'TEAM_DELETED') {
|
||||||
return 1;
|
output.error(err.message);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
throw err;
|
throw err;
|
||||||
@@ -120,7 +63,7 @@ const main = async client => {
|
|||||||
|
|
||||||
export default async client => {
|
export default async client => {
|
||||||
try {
|
try {
|
||||||
await main(client);
|
return await main(client);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
handleError(err);
|
handleError(err);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
@@ -183,7 +126,7 @@ async function run({ output, contextName, currentTeam, client }) {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (out) {
|
if (out) {
|
||||||
console.log(`\n${out}\n`);
|
client.output.print(`\n${out}\n`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -298,19 +241,7 @@ async function run({ output, contextName, currentTeam, client }) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
const [name, parsedValue] = args;
|
const [name, value] = args;
|
||||||
const [originalName, originalValue] = client.argv.slice(-2);
|
|
||||||
|
|
||||||
let value = parsedValue;
|
|
||||||
if (
|
|
||||||
name === originalName &&
|
|
||||||
typeof parsedValue === 'boolean' &&
|
|
||||||
parsedValue !== originalValue
|
|
||||||
) {
|
|
||||||
// Corner case where `mri` transforms the secret value into a boolean because
|
|
||||||
// it starts with a `-` so it thinks its a flag, so we use the original value instead.
|
|
||||||
value = originalValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof value === 'boolean') {
|
if (typeof value === 'boolean') {
|
||||||
const example = chalk.cyan(
|
const example = chalk.cyan(
|
||||||
@@ -340,7 +271,7 @@ async function run({ output, contextName, currentTeam, client }) {
|
|||||||
console.error(
|
console.error(
|
||||||
error('Please specify a valid subcommand: ls | add | rename | rm')
|
error('Please specify a valid subcommand: ls | add | rename | rm')
|
||||||
);
|
);
|
||||||
help();
|
client.output.print(help(secretsCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
77
packages/cli/src/commands/teams/command.ts
Normal file
77
packages/cli/src/commands/teams/command.ts
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
import { Command } from '../help';
|
||||||
|
import { packageName } from '../../util/pkg-name';
|
||||||
|
|
||||||
|
export const teamsCommand: Command = {
|
||||||
|
name: 'teams',
|
||||||
|
description: 'Manage teams under your Vercel account',
|
||||||
|
arguments: [],
|
||||||
|
subcommands: [
|
||||||
|
{
|
||||||
|
name: 'add',
|
||||||
|
description: 'Create a new team',
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'ls',
|
||||||
|
description: "Show all teams you're a part of",
|
||||||
|
arguments: [],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'switch',
|
||||||
|
description: 'Switch to a different team',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'name',
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'invite',
|
||||||
|
description: 'Invite a new member to a team',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'emails',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [],
|
||||||
|
examples: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'next',
|
||||||
|
shorthand: 'N',
|
||||||
|
type: 'string',
|
||||||
|
argument: 'MS',
|
||||||
|
deprecated: false,
|
||||||
|
multi: false,
|
||||||
|
description: 'Show next page of results',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
examples: [
|
||||||
|
{
|
||||||
|
name: "Switch to a team. If your team's url is 'vercel.com/name', then 'name' is the slug. If the slug is omitted, you can choose interactively.",
|
||||||
|
value: `${packageName} teams switch <slug>`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Invite new members (interactively)',
|
||||||
|
value: `${packageName} teams invite`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Invite multiple members simultaneously',
|
||||||
|
value: `${packageName} teams invite abc@vercel.com xyz@vercel.com`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Paginate results, where `1584722256178` is the time in milliseconds since the UNIX epoch.',
|
||||||
|
value: `${packageName} teams ls --next 1584722256178`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,63 +1,12 @@
|
|||||||
import chalk from 'chalk';
|
|
||||||
import error from '../../util/output/error';
|
import error from '../../util/output/error';
|
||||||
import list from './list';
|
import list from './list';
|
||||||
import add from './add';
|
import add from './add';
|
||||||
import change from './switch';
|
import change from './switch';
|
||||||
import invite from './invite';
|
import invite from './invite';
|
||||||
import { packageName, logo } from '../../util/pkg-name';
|
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
|
import { teamsCommand } from './command';
|
||||||
const help = () => {
|
import { help } from '../help';
|
||||||
console.log(`
|
|
||||||
${chalk.bold(`${logo} ${packageName} teams`)} [options] <command>
|
|
||||||
|
|
||||||
${chalk.dim('Commands:')}
|
|
||||||
|
|
||||||
add Create a new team
|
|
||||||
ls Show all teams you're a part of
|
|
||||||
switch [name] Switch to a different team
|
|
||||||
invite [email] Invite a new member to a team
|
|
||||||
|
|
||||||
${chalk.dim('Options:')}
|
|
||||||
|
|
||||||
-h, --help Output usage information
|
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
|
||||||
'FILE'
|
|
||||||
)} Path to the local ${'`vercel.json`'} file
|
|
||||||
-Q ${chalk.bold.underline('DIR')}, --global-config=${chalk.bold.underline(
|
|
||||||
'DIR'
|
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
|
||||||
-d, --debug Debug mode [off]
|
|
||||||
--no-color No color mode [off]
|
|
||||||
-N, --next Show next page of results
|
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Switch to a team
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} switch <slug>`)}
|
|
||||||
|
|
||||||
${chalk.gray(
|
|
||||||
'–'
|
|
||||||
)} If your team's url is 'vercel.com/teams/name', 'name' is the slug
|
|
||||||
${chalk.gray('–')} If the slug is omitted, you can choose interactively
|
|
||||||
|
|
||||||
${chalk.yellow(
|
|
||||||
'NOTE:'
|
|
||||||
)} When you switch, everything you add, list or remove will be scoped that team!
|
|
||||||
|
|
||||||
${chalk.gray('–')} Invite new members (interactively)
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} teams invite`)}
|
|
||||||
|
|
||||||
${chalk.gray('–')} Paginate results, where ${chalk.dim(
|
|
||||||
'`1584722256178`'
|
|
||||||
)} is the time in milliseconds since the UNIX epoch.
|
|
||||||
|
|
||||||
${chalk.cyan(`$ ${packageName} teams ls --next 1584722256178`)}
|
|
||||||
`);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default async (client: Client) => {
|
export default async (client: Client) => {
|
||||||
let subcommand;
|
let subcommand;
|
||||||
@@ -74,7 +23,7 @@ export default async (client: Client) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (argv['--help'] || !subcommand) {
|
if (argv['--help'] || !subcommand) {
|
||||||
help();
|
client.output.print(help(teamsCommand, { columns: client.stderr.columns }));
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -107,7 +56,9 @@ export default async (client: Client) => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
exitCode = 2;
|
exitCode = 2;
|
||||||
help();
|
client.output.print(
|
||||||
|
help(teamsCommand, { columns: client.stderr.columns })
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return exitCode;
|
return exitCode;
|
||||||
|
|||||||
@@ -20,7 +20,8 @@ export async function initCorepack({
|
|||||||
);
|
);
|
||||||
if (pkg instanceof CantParseJSONFile) {
|
if (pkg instanceof CantParseJSONFile) {
|
||||||
console.warn(
|
console.warn(
|
||||||
'Warning: Could not enable corepack because package.json is invalid JSON'
|
'Warning: Could not enable corepack because package.json is invalid JSON',
|
||||||
|
pkg.meta.parseErrorLocation
|
||||||
);
|
);
|
||||||
} else if (!pkg?.packageManager) {
|
} else if (!pkg?.packageManager) {
|
||||||
console.warn(
|
console.warn(
|
||||||
|
|||||||
@@ -1,48 +0,0 @@
|
|||||||
import isWildcardAlias from '../alias/is-wildcard-alias';
|
|
||||||
import { getCertsForCn } from '../certs/get-certs-for-cn';
|
|
||||||
import Client from '../client';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tries to find the "best" alias url.
|
|
||||||
* @param aliasList
|
|
||||||
*/
|
|
||||||
export async function getPreferredPreviewURL(
|
|
||||||
client: Client,
|
|
||||||
aliasList: string[]
|
|
||||||
) {
|
|
||||||
if (aliasList.length === 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* First checks for non public aliases and non wildcard domains.
|
|
||||||
*/
|
|
||||||
const preferredAliases = aliasList.filter(
|
|
||||||
alias =>
|
|
||||||
!alias.endsWith('.now.sh') &&
|
|
||||||
!alias.endsWith('.vercel.app') &&
|
|
||||||
!isWildcardAlias(alias)
|
|
||||||
);
|
|
||||||
for (const alias of preferredAliases) {
|
|
||||||
const certs = await getCertsForCn(client, alias, { limit: 1 }).catch(() => {
|
|
||||||
return null;
|
|
||||||
});
|
|
||||||
if (certs && certs.length > 0) {
|
|
||||||
return { previewUrl: `https://${alias}`, isWildcard: false };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fallback to first alias
|
|
||||||
*/
|
|
||||||
const [firstAlias] = aliasList;
|
|
||||||
if (isWildcardAlias(firstAlias)) {
|
|
||||||
return { previewUrl: firstAlias, isWildcard: true };
|
|
||||||
}
|
|
||||||
|
|
||||||
if (firstAlias.endsWith('.vercel.app') || firstAlias.endsWith('.now.sh')) {
|
|
||||||
return { previewUrl: `https://${firstAlias}`, isWildcard: false };
|
|
||||||
}
|
|
||||||
|
|
||||||
return { previewUrl: `http://${firstAlias}`, isWildcard: false };
|
|
||||||
}
|
|
||||||
@@ -1,20 +1,20 @@
|
|||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
import type Client from '../client';
|
import type Client from '../client';
|
||||||
import type { Deployment } from '@vercel-internals/types';
|
import type { Deployment } from '@vercel-internals/types';
|
||||||
import { getPreferredPreviewURL } from '../../util/deploy/get-preferred-preview-url';
|
|
||||||
import { isDeploying } from '../../util/deploy/is-deploying';
|
import { isDeploying } from '../../util/deploy/is-deploying';
|
||||||
import linkStyle from '../output/link';
|
import linkStyle from '../output/link';
|
||||||
import { prependEmoji, emoji } from '../../util/emoji';
|
import { prependEmoji, emoji } from '../../util/emoji';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prints (to `client.output`) warnings and errors, if any.
|
||||||
|
*/
|
||||||
export async function printDeploymentStatus(
|
export async function printDeploymentStatus(
|
||||||
client: Client,
|
client: Client,
|
||||||
{
|
{
|
||||||
readyState,
|
readyState,
|
||||||
alias: aliasList,
|
|
||||||
aliasError,
|
aliasError,
|
||||||
target,
|
|
||||||
indications,
|
indications,
|
||||||
url: deploymentUrl,
|
|
||||||
aliasWarning,
|
aliasWarning,
|
||||||
}: {
|
}: {
|
||||||
readyState: Deployment['readyState'];
|
readyState: Deployment['readyState'];
|
||||||
@@ -36,7 +36,6 @@ export async function printDeploymentStatus(
|
|||||||
const { output } = client;
|
const { output } = client;
|
||||||
|
|
||||||
indications = indications || [];
|
indications = indications || [];
|
||||||
const isProdDeployment = target === 'production';
|
|
||||||
|
|
||||||
let isStillBuilding = false;
|
let isStillBuilding = false;
|
||||||
if (noWait) {
|
if (noWait) {
|
||||||
@@ -64,30 +63,6 @@ export async function printDeploymentStatus(
|
|||||||
aliasError.message ? `: ${aliasError.message}` : ''
|
aliasError.message ? `: ${aliasError.message}` : ''
|
||||||
}`
|
}`
|
||||||
);
|
);
|
||||||
} else {
|
|
||||||
// print preview/production url
|
|
||||||
let previewUrl: string;
|
|
||||||
// if `noWait` is true, then use the deployment url, not an alias
|
|
||||||
if (!noWait && Array.isArray(aliasList) && aliasList.length > 0) {
|
|
||||||
const previewUrlInfo = await getPreferredPreviewURL(client, aliasList);
|
|
||||||
if (previewUrlInfo) {
|
|
||||||
previewUrl = previewUrlInfo.previewUrl;
|
|
||||||
} else {
|
|
||||||
previewUrl = `https://${deploymentUrl}`;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// fallback to deployment url
|
|
||||||
previewUrl = `https://${deploymentUrl}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
output.print(
|
|
||||||
prependEmoji(
|
|
||||||
`${isProdDeployment ? 'Production' : 'Preview'}: ${chalk.bold(
|
|
||||||
previewUrl
|
|
||||||
)} ${deployStamp()}`,
|
|
||||||
emoji('success')
|
|
||||||
) + `\n`
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (aliasWarning?.message) {
|
if (aliasWarning?.message) {
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import {
|
|||||||
import { Output } from '../output';
|
import { Output } from '../output';
|
||||||
import { progress } from '../output/progress';
|
import { progress } from '../output/progress';
|
||||||
import Now from '../../util';
|
import Now from '../../util';
|
||||||
import type { Org } from '@vercel-internals/types';
|
import type { Deployment, Org } from '@vercel-internals/types';
|
||||||
import ua from '../ua';
|
import ua from '../ua';
|
||||||
import { linkFolderToProject } from '../projects/link';
|
import { linkFolderToProject } from '../projects/link';
|
||||||
import { prependEmoji, emoji } from '../emoji';
|
import { prependEmoji, emoji } from '../emoji';
|
||||||
@@ -17,9 +17,13 @@ import type { Agent } from 'http';
|
|||||||
|
|
||||||
function printInspectUrl(
|
function printInspectUrl(
|
||||||
output: Output,
|
output: Output,
|
||||||
inspectorUrl: string,
|
inspectorUrl: string | null | undefined,
|
||||||
deployStamp: () => string
|
deployStamp: () => string
|
||||||
) {
|
) {
|
||||||
|
if (!inspectorUrl) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
output.print(
|
output.print(
|
||||||
prependEmoji(
|
prependEmoji(
|
||||||
`Inspect: ${chalk.bold(inspectorUrl)} ${deployStamp()}`,
|
`Inspect: ${chalk.bold(inspectorUrl)} ${deployStamp()}`,
|
||||||
@@ -162,33 +166,47 @@ export default async function processDeployment({
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (event.type === 'created') {
|
if (event.type === 'created') {
|
||||||
|
const deployment: Deployment = event.payload;
|
||||||
|
|
||||||
await linkFolderToProject(
|
await linkFolderToProject(
|
||||||
client,
|
client,
|
||||||
cwd,
|
cwd,
|
||||||
{
|
{
|
||||||
orgId: org.id,
|
orgId: org.id,
|
||||||
projectId: event.payload.projectId,
|
projectId: deployment.projectId!,
|
||||||
},
|
},
|
||||||
projectName,
|
projectName,
|
||||||
org.slug
|
org.slug
|
||||||
);
|
);
|
||||||
|
|
||||||
now.url = event.payload.url;
|
now.url = deployment.url;
|
||||||
|
|
||||||
output.stopSpinner();
|
output.stopSpinner();
|
||||||
|
|
||||||
printInspectUrl(output, event.payload.inspectorUrl, deployStamp);
|
printInspectUrl(output, deployment.inspectorUrl, deployStamp);
|
||||||
|
|
||||||
|
const isProdDeployment = requestBody.target === 'production';
|
||||||
|
const previewUrl = `https://${deployment.url}`;
|
||||||
|
|
||||||
|
output.print(
|
||||||
|
prependEmoji(
|
||||||
|
`${isProdDeployment ? 'Production' : 'Preview'}: ${chalk.bold(
|
||||||
|
previewUrl
|
||||||
|
)} ${deployStamp()}`,
|
||||||
|
emoji('success')
|
||||||
|
) + `\n`
|
||||||
|
);
|
||||||
|
|
||||||
if (quiet) {
|
if (quiet) {
|
||||||
process.stdout.write(`https://${event.payload.url}`);
|
process.stdout.write(`https://${event.payload.url}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (noWait) {
|
if (noWait) {
|
||||||
return event.payload;
|
return deployment;
|
||||||
}
|
}
|
||||||
|
|
||||||
output.spinner(
|
output.spinner(
|
||||||
event.payload.readyState === 'QUEUED' ? 'Queued' : 'Building',
|
deployment.readyState === 'QUEUED' ? 'Queued' : 'Building',
|
||||||
0
|
0
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -243,6 +261,7 @@ export default async function processDeployment({
|
|||||||
|
|
||||||
// Handle alias-assigned event
|
// Handle alias-assigned event
|
||||||
if (event.type === 'alias-assigned') {
|
if (event.type === 'alias-assigned') {
|
||||||
|
output.stopSpinner();
|
||||||
event.payload.indications = indications;
|
event.payload.indications = indications;
|
||||||
return event.payload;
|
return event.payload;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ import isPortReachable from 'is-port-reachable';
|
|||||||
import deepEqual from 'fast-deep-equal';
|
import deepEqual from 'fast-deep-equal';
|
||||||
import npa from 'npm-package-arg';
|
import npa from 'npm-package-arg';
|
||||||
import type { ChildProcess } from 'child_process';
|
import type { ChildProcess } from 'child_process';
|
||||||
|
import JSONparse from 'json-parse-better-errors';
|
||||||
|
|
||||||
import { getVercelIgnore, fileNameSymbol } from '@vercel/client';
|
import { getVercelIgnore, fileNameSymbol } from '@vercel/client';
|
||||||
import {
|
import {
|
||||||
@@ -726,7 +727,7 @@ export default class DevServer {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const raw = await fs.readFile(abs, 'utf8');
|
const raw = await fs.readFile(abs, 'utf8');
|
||||||
const parsed: WithFileNameSymbol<T> = JSON.parse(raw);
|
const parsed: WithFileNameSymbol<T> = JSONparse(raw);
|
||||||
parsed[fileNameSymbol] = rel;
|
parsed[fileNameSymbol] = rel;
|
||||||
return parsed;
|
return parsed;
|
||||||
} catch (err: unknown) {
|
} catch (err: unknown) {
|
||||||
|
|||||||
@@ -668,13 +668,14 @@ export class CertMissing extends NowError<'ALIAS_IN_USE', { domain: string }> {
|
|||||||
|
|
||||||
export class CantParseJSONFile extends NowError<
|
export class CantParseJSONFile extends NowError<
|
||||||
'CANT_PARSE_JSON_FILE',
|
'CANT_PARSE_JSON_FILE',
|
||||||
{ file: string }
|
{ file: string; parseErrorLocation: string }
|
||||||
> {
|
> {
|
||||||
constructor(file: string) {
|
constructor(file: string, parseErrorLocation: string) {
|
||||||
|
const message = `Can't parse json file ${file}: ${parseErrorLocation}`;
|
||||||
super({
|
super({
|
||||||
code: 'CANT_PARSE_JSON_FILE',
|
code: 'CANT_PARSE_JSON_FILE',
|
||||||
meta: { file },
|
meta: { file, parseErrorLocation },
|
||||||
message: `Can't parse json file`,
|
message,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
export default (code?: number) =>
|
|
||||||
new Promise(() => {
|
|
||||||
// We give stdout some time to flush out
|
|
||||||
// because there's a node bug where
|
|
||||||
// stdout writes are asynchronous
|
|
||||||
// https://github.com/nodejs/node/issues/6456
|
|
||||||
setTimeout(() => process.exit(code || 0), 100);
|
|
||||||
});
|
|
||||||
@@ -1,5 +1,7 @@
|
|||||||
import fs from 'fs-extra';
|
import fs from 'fs-extra';
|
||||||
import { CantParseJSONFile } from './errors-ts';
|
import { CantParseJSONFile } from './errors-ts';
|
||||||
|
import JSONparse from 'json-parse-better-errors';
|
||||||
|
import { errorToString } from '@vercel/error-utils';
|
||||||
|
|
||||||
export default async function readJSONFile<T>(
|
export default async function readJSONFile<T>(
|
||||||
file: string
|
file: string
|
||||||
@@ -10,10 +12,10 @@ export default async function readJSONFile<T>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const json = JSON.parse(content);
|
const json = JSONparse(content);
|
||||||
return json;
|
return json;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return new CantParseJSONFile(file);
|
return new CantParseJSONFile(file, errorToString(error));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
2
packages/cli/test/integration-1.test.ts
vendored
2
packages/cli/test/integration-1.test.ts
vendored
@@ -292,7 +292,7 @@ test('[vc dev] should print help from `vc develop --help`', async () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
expect(exitCode, formatOutput({ stdout, stderr })).toBe(2);
|
expect(exitCode, formatOutput({ stdout, stderr })).toBe(2);
|
||||||
expect(stdout).toMatch(/▲ vercel dev/gm);
|
expect(stderr).toMatch(/▲ vercel dev/gm);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('default command should deploy directory', async () => {
|
test('default command should deploy directory', async () => {
|
||||||
|
|||||||
6
packages/cli/test/integration-2.test.ts
vendored
6
packages/cli/test/integration-2.test.ts
vendored
@@ -735,6 +735,12 @@ test('deploys with only vercel.json and README.md', async () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
expect(exitCode, formatOutput({ stdout, stderr })).toBe(0);
|
expect(exitCode, formatOutput({ stdout, stderr })).toBe(0);
|
||||||
|
|
||||||
|
// assert timing order of showing URLs vs status updates
|
||||||
|
expect(stderr).toMatch(
|
||||||
|
/Inspect.*\nPreview.*\nQueued.*\nBuilding.*\nCompleting/
|
||||||
|
);
|
||||||
|
|
||||||
const { host } = new URL(stdout);
|
const { host } = new URL(stdout);
|
||||||
const res = await fetch(`https://${host}/README.md`);
|
const res = await fetch(`https://${host}/README.md`);
|
||||||
const text = await res.text();
|
const text = await res.text();
|
||||||
|
|||||||
29
packages/cli/test/mocks/secrets.ts
Normal file
29
packages/cli/test/mocks/secrets.ts
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import chance from 'chance';
|
||||||
|
import { client } from './client';
|
||||||
|
|
||||||
|
export function useSecrets({
|
||||||
|
name,
|
||||||
|
created,
|
||||||
|
}: {
|
||||||
|
name: string;
|
||||||
|
created: number;
|
||||||
|
}) {
|
||||||
|
const secret = {
|
||||||
|
uid: chance().guid(),
|
||||||
|
name: name || chance().name(),
|
||||||
|
created: created || chance().timestamp(),
|
||||||
|
};
|
||||||
|
|
||||||
|
client.scenario.get('/v3/now/secrets', (_req, res) => {
|
||||||
|
res.json({
|
||||||
|
secrets: [secret],
|
||||||
|
pagination: {
|
||||||
|
count: 1,
|
||||||
|
next: 0,
|
||||||
|
prev: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return secret;
|
||||||
|
}
|
||||||
37
packages/cli/test/unit/commands/secrets.test.ts
Normal file
37
packages/cli/test/unit/commands/secrets.test.ts
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import { client } from '../../mocks/client';
|
||||||
|
import secrets from '../../../src/commands/secrets';
|
||||||
|
import { useSecrets } from '../../mocks/secrets';
|
||||||
|
import { useUser } from '../../mocks/user';
|
||||||
|
import { useTeams } from '../../mocks/team';
|
||||||
|
import ms from 'ms';
|
||||||
|
|
||||||
|
describe('secrets', () => {
|
||||||
|
it('errors when no subcommand is provided', async () => {
|
||||||
|
client.setArgv('secrets');
|
||||||
|
const exitCode = await secrets(client);
|
||||||
|
|
||||||
|
expect(exitCode).toEqual(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('lists secrets with ls subcommand', async () => {
|
||||||
|
useUser();
|
||||||
|
useTeams('team_dummy');
|
||||||
|
const name = 'secret-api-password';
|
||||||
|
const created = 1519555701;
|
||||||
|
useSecrets({ name, created });
|
||||||
|
|
||||||
|
client.setArgv('secrets', 'ls');
|
||||||
|
await secrets(client);
|
||||||
|
|
||||||
|
const timeAgo = `${ms(
|
||||||
|
new Date().getTime() - new Date(created).getTime()
|
||||||
|
)} ago`;
|
||||||
|
await expect(client.stderr).toOutput(
|
||||||
|
'> NOTE: The `vercel env ls` command is recommended instead of `vercel secret ls`'
|
||||||
|
);
|
||||||
|
|
||||||
|
const output = client.stderr.read();
|
||||||
|
await expect(output).toMatch(name);
|
||||||
|
await expect(output).toMatch(timeAgo);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,5 +1,19 @@
|
|||||||
# @vercel/client
|
# @vercel/client
|
||||||
|
|
||||||
|
## 13.0.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`5609a1187`](https://github.com/vercel/vercel/commit/5609a1187be9d6cf8d5f16825690c5ea72f17dc5), [`1b4de4a98`](https://github.com/vercel/vercel/commit/1b4de4a986f7a612aac834ebae3ec7bb9e9b8cf8)]:
|
||||||
|
- @vercel/build-utils@7.1.1
|
||||||
|
|
||||||
|
## 13.0.1
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`9e3827c78`](https://github.com/vercel/vercel/commit/9e3827c785e1bc45f2bed421132167381481770f)]:
|
||||||
|
- @vercel/build-utils@7.1.0
|
||||||
|
|
||||||
## 13.0.0
|
## 13.0.0
|
||||||
|
|
||||||
### Major Changes
|
### Major Changes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vercel/client",
|
"name": "@vercel/client",
|
||||||
"version": "13.0.0",
|
"version": "13.0.2",
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
"typings": "dist/index.d.ts",
|
"typings": "dist/index.d.ts",
|
||||||
"homepage": "https://vercel.com",
|
"homepage": "https://vercel.com",
|
||||||
@@ -14,7 +14,7 @@
|
|||||||
"directory": "packages/client"
|
"directory": "packages/client"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsc",
|
"build": "node ../../utils/build.mjs",
|
||||||
"test-e2e": "pnpm test tests/create-deployment.test.ts tests/create-legacy-deployment.test.ts tests/paths.test.ts",
|
"test-e2e": "pnpm test tests/create-deployment.test.ts tests/create-legacy-deployment.test.ts tests/paths.test.ts",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
||||||
"test-unit": "pnpm test tests/unit.*test.*"
|
"test-unit": "pnpm test tests/unit.*test.*"
|
||||||
@@ -36,7 +36,7 @@
|
|||||||
"typescript": "4.9.5"
|
"typescript": "4.9.5"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/routing-utils": "3.0.0",
|
"@vercel/routing-utils": "3.0.0",
|
||||||
"@zeit/fetch": "5.2.0",
|
"@zeit/fetch": "5.2.0",
|
||||||
"async-retry": "1.2.3",
|
"async-retry": "1.2.3",
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"declaration": true,
|
"declaration": true,
|
||||||
|
"emitDeclarationOnly": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"lib": ["ES2021"],
|
"lib": ["ES2021"],
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
"directory": "packages/error-utils"
|
"directory": "packages/error-utils"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsc",
|
"build": "node ../../utils/build.mjs",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --coverage --env node --verbose",
|
"test": "jest --reporters=default --reporters=jest-junit --coverage --env node --verbose",
|
||||||
"test-unit": "pnpm test"
|
"test-unit": "pnpm test"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"declaration": true,
|
"declaration": true,
|
||||||
"declarationMap": true,
|
"emitDeclarationOnly": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"lib": ["ES2021"],
|
"lib": ["ES2021"],
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
},
|
},
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsc",
|
"build": "node ../../utils/build.mjs",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
||||||
"test-unit": "pnpm test"
|
"test-unit": "pnpm test"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"declaration": true,
|
"declaration": true,
|
||||||
|
"emitDeclarationOnly": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"lib": ["ES2021"],
|
"lib": ["ES2021"],
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
|
|||||||
@@ -1,5 +1,11 @@
|
|||||||
# @vercel/fs-detectors
|
# @vercel/fs-detectors
|
||||||
|
|
||||||
|
## 5.0.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated semver dependency ([#10411](https://github.com/vercel/vercel/pull/10411))
|
||||||
|
|
||||||
## 5.0.1
|
## 5.0.1
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vercel/fs-detectors",
|
"name": "@vercel/fs-detectors",
|
||||||
"version": "5.0.1",
|
"version": "5.0.2",
|
||||||
"description": "Vercel filesystem detectors",
|
"description": "Vercel filesystem detectors",
|
||||||
"main": "./dist/index.js",
|
"main": "./dist/index.js",
|
||||||
"types": "./dist/index.d.ts",
|
"types": "./dist/index.d.ts",
|
||||||
@@ -14,7 +14,7 @@
|
|||||||
},
|
},
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsc",
|
"build": "node ../../utils/build.mjs",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --runInBand --bail",
|
||||||
"test-unit": "pnpm test test/unit.*test.*",
|
"test-unit": "pnpm test test/unit.*test.*",
|
||||||
"test-e2e": "pnpm test test/integration.test.ts"
|
"test-e2e": "pnpm test test/integration.test.ts"
|
||||||
@@ -27,7 +27,7 @@
|
|||||||
"js-yaml": "4.1.0",
|
"js-yaml": "4.1.0",
|
||||||
"json5": "2.2.2",
|
"json5": "2.2.2",
|
||||||
"minimatch": "3.1.2",
|
"minimatch": "3.1.2",
|
||||||
"semver": "6.1.1"
|
"semver": "6.3.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/glob": "7.2.0",
|
"@types/glob": "7.2.0",
|
||||||
@@ -36,7 +36,7 @@
|
|||||||
"@types/minimatch": "3.0.5",
|
"@types/minimatch": "3.0.5",
|
||||||
"@types/node": "14.18.33",
|
"@types/node": "14.18.33",
|
||||||
"@types/semver": "7.3.10",
|
"@types/semver": "7.3.10",
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"jest-junit": "16.0.0",
|
"jest-junit": "16.0.0",
|
||||||
"typescript": "4.9.5"
|
"typescript": "4.9.5"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"declaration": true,
|
"declaration": true,
|
||||||
|
"emitDeclarationOnly": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"lib": ["ES2021"],
|
"lib": ["ES2021"],
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
|
|||||||
@@ -1,5 +1,21 @@
|
|||||||
# @vercel/gatsby-plugin-vercel-builder
|
# @vercel/gatsby-plugin-vercel-builder
|
||||||
|
|
||||||
|
## 2.0.4
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`5609a1187`](https://github.com/vercel/vercel/commit/5609a1187be9d6cf8d5f16825690c5ea72f17dc5), [`1b4de4a98`](https://github.com/vercel/vercel/commit/1b4de4a986f7a612aac834ebae3ec7bb9e9b8cf8), [`fc1e13c09`](https://github.com/vercel/vercel/commit/fc1e13c09928c654410b373fc1775c2b63c6ef4a)]:
|
||||||
|
- @vercel/build-utils@7.1.1
|
||||||
|
- @vercel/node@3.0.4
|
||||||
|
|
||||||
|
## 2.0.3
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`9e3827c78`](https://github.com/vercel/vercel/commit/9e3827c785e1bc45f2bed421132167381481770f)]:
|
||||||
|
- @vercel/build-utils@7.1.0
|
||||||
|
- @vercel/node@3.0.3
|
||||||
|
|
||||||
## 2.0.2
|
## 2.0.2
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vercel/gatsby-plugin-vercel-builder",
|
"name": "@vercel/gatsby-plugin-vercel-builder",
|
||||||
"version": "2.0.2",
|
"version": "2.0.4",
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
"files": [
|
"files": [
|
||||||
"dist",
|
"dist",
|
||||||
@@ -20,8 +20,8 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sinclair/typebox": "0.25.24",
|
"@sinclair/typebox": "0.25.24",
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/node": "3.0.2",
|
"@vercel/node": "3.0.4",
|
||||||
"@vercel/routing-utils": "3.0.0",
|
"@vercel/routing-utils": "3.0.0",
|
||||||
"esbuild": "0.14.47",
|
"esbuild": "0.14.47",
|
||||||
"etag": "1.8.1",
|
"etag": "1.8.1",
|
||||||
|
|||||||
@@ -27,7 +27,7 @@
|
|||||||
"@types/node-fetch": "^2.3.0",
|
"@types/node-fetch": "^2.3.0",
|
||||||
"@types/tar": "^4.0.0",
|
"@types/tar": "^4.0.0",
|
||||||
"@types/yauzl-promise": "2.1.0",
|
"@types/yauzl-promise": "2.1.0",
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/ncc": "0.24.0",
|
"@vercel/ncc": "0.24.0",
|
||||||
"async-retry": "1.3.1",
|
"async-retry": "1.3.1",
|
||||||
"execa": "^1.0.0",
|
"execa": "^1.0.0",
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
const execa = require('execa');
|
|
||||||
const { remove } = require('fs-extra');
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
await remove('dist');
|
|
||||||
await execa('tsc', [], { stdio: 'inherit' });
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(err => {
|
|
||||||
console.error(err);
|
|
||||||
process.exit(1);
|
|
||||||
});
|
|
||||||
@@ -10,7 +10,7 @@
|
|||||||
"directory": "packages/hydrogen"
|
"directory": "packages/hydrogen"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "node build.js",
|
"build": "node ../../utils/build.mjs",
|
||||||
"test-e2e": "pnpm test test/test.js",
|
"test-e2e": "pnpm test test/test.js",
|
||||||
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --bail --runInBand"
|
"test": "jest --reporters=default --reporters=jest-junit --env node --verbose --bail --runInBand"
|
||||||
},
|
},
|
||||||
@@ -21,7 +21,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/jest": "27.5.1",
|
"@types/jest": "27.5.1",
|
||||||
"@types/node": "14.18.33",
|
"@types/node": "14.18.33",
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/static-config": "3.0.0",
|
"@vercel/static-config": "3.0.0",
|
||||||
"execa": "3.2.0",
|
"execa": "3.2.0",
|
||||||
"fs-extra": "11.1.0",
|
"fs-extra": "11.1.0",
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"declaration": true,
|
"declaration": true,
|
||||||
|
"emitDeclarationOnly": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"lib": ["ES2021"],
|
"lib": ["ES2021"],
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
|
|||||||
@@ -1,5 +1,19 @@
|
|||||||
# @vercel/next
|
# @vercel/next
|
||||||
|
|
||||||
|
## 4.0.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Fix Next.js with `basePath` + Edge runtime + App Router on a top level `page.jsx` ([#10465](https://github.com/vercel/vercel/pull/10465))
|
||||||
|
|
||||||
|
- Updated semver dependency ([#10411](https://github.com/vercel/vercel/pull/10411))
|
||||||
|
|
||||||
|
- Fix RSC rewrite behavior ([#10415](https://github.com/vercel/vercel/pull/10415))
|
||||||
|
|
||||||
|
- fix ENOENT on /404.html when `fallback: false` w/ `basePath` ([#10473](https://github.com/vercel/vercel/pull/10473))
|
||||||
|
|
||||||
|
- fix 404 enoent for i18n ([#10416](https://github.com/vercel/vercel/pull/10416))
|
||||||
|
|
||||||
## 4.0.1
|
## 4.0.1
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vercel/next",
|
"name": "@vercel/next",
|
||||||
"version": "4.0.1",
|
"version": "4.0.2",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"main": "./dist/index",
|
"main": "./dist/index",
|
||||||
"homepage": "https://vercel.com/docs/runtimes#official-runtimes/next-js",
|
"homepage": "https://vercel.com/docs/runtimes#official-runtimes/next-js",
|
||||||
@@ -35,7 +35,7 @@
|
|||||||
"@types/semver": "6.0.0",
|
"@types/semver": "6.0.0",
|
||||||
"@types/text-table": "0.2.1",
|
"@types/text-table": "0.2.1",
|
||||||
"@types/webpack-sources": "3.2.0",
|
"@types/webpack-sources": "3.2.0",
|
||||||
"@vercel/build-utils": "7.0.0",
|
"@vercel/build-utils": "7.1.1",
|
||||||
"@vercel/nft": "0.22.5",
|
"@vercel/nft": "0.22.5",
|
||||||
"@vercel/routing-utils": "3.0.0",
|
"@vercel/routing-utils": "3.0.0",
|
||||||
"async-sema": "3.0.1",
|
"async-sema": "3.0.1",
|
||||||
@@ -54,7 +54,7 @@
|
|||||||
"ndjson": "2.0.0",
|
"ndjson": "2.0.0",
|
||||||
"pretty-bytes": "5.3.0",
|
"pretty-bytes": "5.3.0",
|
||||||
"resolve-from": "5.0.0",
|
"resolve-from": "5.0.0",
|
||||||
"semver": "6.1.1",
|
"semver": "6.3.1",
|
||||||
"set-cookie-parser": "2.4.6",
|
"set-cookie-parser": "2.4.6",
|
||||||
"source-map": "0.7.3",
|
"source-map": "0.7.3",
|
||||||
"test-listen": "1.1.0",
|
"test-listen": "1.1.0",
|
||||||
|
|||||||
@@ -91,6 +91,7 @@ import {
|
|||||||
getOperationType,
|
getOperationType,
|
||||||
isApiPage,
|
isApiPage,
|
||||||
getFunctionsConfigManifest,
|
getFunctionsConfigManifest,
|
||||||
|
normalizeEdgeFunctionPath,
|
||||||
} from './utils';
|
} from './utils';
|
||||||
|
|
||||||
export const version = 2;
|
export const version = 2;
|
||||||
@@ -1320,11 +1321,7 @@ export const build: BuildV2 = async ({
|
|||||||
const localePrefixed404 = !!(
|
const localePrefixed404 = !!(
|
||||||
routesManifest.i18n &&
|
routesManifest.i18n &&
|
||||||
originalStaticPages[
|
originalStaticPages[
|
||||||
path.posix.join(
|
path.posix.join('.', routesManifest.i18n.defaultLocale, '404.html')
|
||||||
entryDirectory,
|
|
||||||
routesManifest.i18n.defaultLocale,
|
|
||||||
'404.html'
|
|
||||||
)
|
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -2720,7 +2717,15 @@ async function getServerlessPages(params: {
|
|||||||
for (const edgeFunctionFile of Object.keys(
|
for (const edgeFunctionFile of Object.keys(
|
||||||
middlewareManifest?.functions ?? {}
|
middlewareManifest?.functions ?? {}
|
||||||
)) {
|
)) {
|
||||||
const edgePath = (edgeFunctionFile.slice(1) || 'index') + '.js';
|
let edgePath =
|
||||||
|
middlewareManifest?.functions?.[edgeFunctionFile].name ||
|
||||||
|
edgeFunctionFile;
|
||||||
|
|
||||||
|
edgePath = normalizeEdgeFunctionPath(
|
||||||
|
edgePath,
|
||||||
|
params.appPathRoutesManifest || {}
|
||||||
|
);
|
||||||
|
edgePath = (edgePath || 'index') + '.js';
|
||||||
delete normalizedAppPaths[edgePath];
|
delete normalizedAppPaths[edgePath];
|
||||||
delete pages[edgePath];
|
delete pages[edgePath];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -437,11 +437,19 @@ export async function serverBuild({
|
|||||||
|
|
||||||
if (i18n) {
|
if (i18n) {
|
||||||
for (const locale of i18n.locales) {
|
for (const locale of i18n.locales) {
|
||||||
const static404File =
|
let static404File =
|
||||||
staticPages[path.posix.join(entryDirectory, locale, '/404')] ||
|
staticPages[path.posix.join(entryDirectory, locale, '/404')];
|
||||||
new FileFsRef({
|
|
||||||
|
if (!static404File) {
|
||||||
|
static404File = new FileFsRef({
|
||||||
fsPath: path.join(pagesDir, locale, '/404.html'),
|
fsPath: path.join(pagesDir, locale, '/404.html'),
|
||||||
});
|
});
|
||||||
|
if (!fs.existsSync(static404File.fsPath)) {
|
||||||
|
static404File = new FileFsRef({
|
||||||
|
fsPath: path.join(pagesDir, '/404.html'),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
requiredFiles[path.relative(baseDir, static404File.fsPath)] =
|
requiredFiles[path.relative(baseDir, static404File.fsPath)] =
|
||||||
static404File;
|
static404File;
|
||||||
}
|
}
|
||||||
@@ -1066,8 +1074,7 @@ export async function serverBuild({
|
|||||||
canUsePreviewMode,
|
canUsePreviewMode,
|
||||||
prerenderManifest.bypassToken || '',
|
prerenderManifest.bypassToken || '',
|
||||||
true,
|
true,
|
||||||
middleware.dynamicRouteMap,
|
middleware.dynamicRouteMap
|
||||||
inversedAppPathManifest
|
|
||||||
).then(arr =>
|
).then(arr =>
|
||||||
localizeDynamicRoutes(
|
localizeDynamicRoutes(
|
||||||
arr,
|
arr,
|
||||||
@@ -1082,6 +1089,32 @@ export async function serverBuild({
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const pagesPlaceholderRscEntries: Record<string, FileBlob> = {};
|
||||||
|
|
||||||
|
if (appDir) {
|
||||||
|
// since we attempt to rewrite all paths to an .rsc variant,
|
||||||
|
// we need to create dummy rsc outputs for all pages entries
|
||||||
|
// this is so that an RSC request to a `pages` entry will match
|
||||||
|
// rather than falling back to a catchall `app` entry
|
||||||
|
// on the nextjs side, invalid RSC response payloads will correctly trigger an mpa navigation
|
||||||
|
const pagesManifest = path.join(
|
||||||
|
entryPath,
|
||||||
|
outputDirectory,
|
||||||
|
`server/pages-manifest.json`
|
||||||
|
);
|
||||||
|
|
||||||
|
const pagesData = await fs.readJSON(pagesManifest);
|
||||||
|
const pagesEntries = Object.keys(pagesData);
|
||||||
|
|
||||||
|
for (const page of pagesEntries) {
|
||||||
|
const pathName = page.startsWith('/') ? page.slice(1) : page;
|
||||||
|
pagesPlaceholderRscEntries[`${pathName}.rsc`] = new FileBlob({
|
||||||
|
data: '{}',
|
||||||
|
contentType: 'application/json',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const { staticFiles, publicDirectoryFiles, staticDirectoryFiles } =
|
const { staticFiles, publicDirectoryFiles, staticDirectoryFiles } =
|
||||||
await getStaticFiles(entryPath, entryDirectory, outputDirectory);
|
await getStaticFiles(entryPath, entryDirectory, outputDirectory);
|
||||||
|
|
||||||
@@ -1212,10 +1245,9 @@ export async function serverBuild({
|
|||||||
if (ogRoute.endsWith('/route')) {
|
if (ogRoute.endsWith('/route')) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
route = path.posix.join(
|
route = normalizeIndexOutput(
|
||||||
'./',
|
path.posix.join('./', entryDirectory, route === '/' ? '/index' : route),
|
||||||
entryDirectory,
|
true
|
||||||
route === '/' ? '/index' : route
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (lambdas[route]) {
|
if (lambdas[route]) {
|
||||||
@@ -1241,6 +1273,7 @@ export async function serverBuild({
|
|||||||
...publicDirectoryFiles,
|
...publicDirectoryFiles,
|
||||||
...lambdas,
|
...lambdas,
|
||||||
...appRscPrefetches,
|
...appRscPrefetches,
|
||||||
|
...pagesPlaceholderRscEntries,
|
||||||
// Prerenders may override Lambdas -- this is an intentional behavior.
|
// Prerenders may override Lambdas -- this is an intentional behavior.
|
||||||
...prerenders,
|
...prerenders,
|
||||||
...staticPages,
|
...staticPages,
|
||||||
@@ -1624,72 +1657,22 @@ export async function serverBuild({
|
|||||||
]
|
]
|
||||||
: []),
|
: []),
|
||||||
|
|
||||||
...(appDir
|
|
||||||
? [
|
|
||||||
// check routes that end in `.rsc` to see if a page with the resulting name (sans-.rsc) exists in the filesystem
|
|
||||||
// if so, we want to match that page instead. (This matters when prefetching a pages route while on an appdir route)
|
|
||||||
{
|
|
||||||
src: `^${path.posix.join('/', entryDirectory, '/(.*)\\.rsc$')}`,
|
|
||||||
dest: path.posix.join('/', entryDirectory, '/$1'),
|
|
||||||
has: [
|
|
||||||
{
|
|
||||||
type: 'header',
|
|
||||||
key: rscHeader,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
...(rscPrefetchHeader
|
|
||||||
? {
|
|
||||||
missing: [
|
|
||||||
{
|
|
||||||
type: 'header',
|
|
||||||
key: rscPrefetchHeader,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}
|
|
||||||
: {}),
|
|
||||||
check: true,
|
|
||||||
} as Route,
|
|
||||||
]
|
|
||||||
: []),
|
|
||||||
|
|
||||||
// These need to come before handle: miss or else they are grouped
|
// These need to come before handle: miss or else they are grouped
|
||||||
// with that routing section
|
// with that routing section
|
||||||
...afterFilesRewrites,
|
...afterFilesRewrites,
|
||||||
|
|
||||||
...(appDir
|
|
||||||
? [
|
|
||||||
// rewrite route back to `.rsc`, but skip checking fs
|
|
||||||
{
|
|
||||||
src: `^${path.posix.join(
|
|
||||||
'/',
|
|
||||||
entryDirectory,
|
|
||||||
'/((?!.+\\.rsc).+?)(?:/)?$'
|
|
||||||
)}`,
|
|
||||||
has: [
|
|
||||||
{
|
|
||||||
type: 'header',
|
|
||||||
key: rscHeader,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
dest: path.posix.join('/', entryDirectory, '/$1.rsc'),
|
|
||||||
headers: { vary: rscVaryHeader },
|
|
||||||
continue: true,
|
|
||||||
override: true,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
: []),
|
|
||||||
|
|
||||||
// make sure 404 page is used when a directory is matched without
|
|
||||||
// an index page
|
|
||||||
{ handle: 'resource' },
|
{ handle: 'resource' },
|
||||||
|
|
||||||
...fallbackRewrites,
|
...fallbackRewrites,
|
||||||
|
|
||||||
|
// make sure 404 page is used when a directory is matched without
|
||||||
|
// an index page
|
||||||
{ src: path.posix.join('/', entryDirectory, '.*'), status: 404 },
|
{ src: path.posix.join('/', entryDirectory, '.*'), status: 404 },
|
||||||
|
|
||||||
|
{ handle: 'miss' },
|
||||||
|
|
||||||
// We need to make sure to 404 for /_next after handle: miss since
|
// We need to make sure to 404 for /_next after handle: miss since
|
||||||
// handle: miss is called before rewrites and to prevent rewriting /_next
|
// handle: miss is called before rewrites and to prevent rewriting /_next
|
||||||
{ handle: 'miss' },
|
|
||||||
{
|
{
|
||||||
src: path.posix.join(
|
src: path.posix.join(
|
||||||
'/',
|
'/',
|
||||||
|
|||||||
@@ -305,8 +305,7 @@ export async function getDynamicRoutes(
|
|||||||
canUsePreviewMode?: boolean,
|
canUsePreviewMode?: boolean,
|
||||||
bypassToken?: string,
|
bypassToken?: string,
|
||||||
isServerMode?: boolean,
|
isServerMode?: boolean,
|
||||||
dynamicMiddlewareRouteMap?: Map<string, RouteWithSrc>,
|
dynamicMiddlewareRouteMap?: Map<string, RouteWithSrc>
|
||||||
appPathRoutesManifest?: Record<string, string>
|
|
||||||
): Promise<RouteWithSrc[]> {
|
): Promise<RouteWithSrc[]> {
|
||||||
if (routesManifest) {
|
if (routesManifest) {
|
||||||
switch (routesManifest.version) {
|
switch (routesManifest.version) {
|
||||||
@@ -379,17 +378,15 @@ export async function getDynamicRoutes(
|
|||||||
},
|
},
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
routes.push({
|
||||||
|
...route,
|
||||||
|
src: route.src.replace(
|
||||||
|
new RegExp(escapeStringRegexp('(?:/)?$')),
|
||||||
|
'(?:\\.rsc)(?:/)?$'
|
||||||
|
),
|
||||||
|
dest: route.dest?.replace(/($|\?)/, '.rsc$1'),
|
||||||
|
});
|
||||||
|
|
||||||
if (appPathRoutesManifest?.[page]) {
|
|
||||||
routes.push({
|
|
||||||
...route,
|
|
||||||
src: route.src.replace(
|
|
||||||
new RegExp(escapeStringRegexp('(?:/)?$')),
|
|
||||||
'(?:\\.rsc)(?:/)?$'
|
|
||||||
),
|
|
||||||
dest: route.dest?.replace(/($|\?)/, '.rsc$1'),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
routes.push(route);
|
routes.push(route);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -2347,7 +2344,7 @@ export function normalizeIndexOutput(
|
|||||||
outputName: string,
|
outputName: string,
|
||||||
isServerMode: boolean
|
isServerMode: boolean
|
||||||
) {
|
) {
|
||||||
if (outputName !== '/index' && isServerMode) {
|
if (outputName !== 'index' && outputName !== '/index' && isServerMode) {
|
||||||
return outputName.replace(/\/index$/, '');
|
return outputName.replace(/\/index$/, '');
|
||||||
}
|
}
|
||||||
return outputName;
|
return outputName;
|
||||||
@@ -2524,6 +2521,29 @@ function normalizeRegions(regions: Regions): undefined | string | string[] {
|
|||||||
return newRegions;
|
return newRegions;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function normalizeEdgeFunctionPath(
|
||||||
|
shortPath: string,
|
||||||
|
appPathRoutesManifest: Record<string, string>
|
||||||
|
) {
|
||||||
|
if (
|
||||||
|
shortPath.startsWith('app/') &&
|
||||||
|
(shortPath.endsWith('/page') ||
|
||||||
|
shortPath.endsWith('/route') ||
|
||||||
|
shortPath === 'app/_not-found')
|
||||||
|
) {
|
||||||
|
const ogRoute = shortPath.replace(/^app\//, '/');
|
||||||
|
shortPath = (
|
||||||
|
appPathRoutesManifest[ogRoute] ||
|
||||||
|
shortPath.replace(/(^|\/)(page|route)$/, '')
|
||||||
|
).replace(/^\//, '');
|
||||||
|
|
||||||
|
if (!shortPath || shortPath === '/') {
|
||||||
|
shortPath = 'index';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return shortPath;
|
||||||
|
}
|
||||||
|
|
||||||
export async function getMiddlewareBundle({
|
export async function getMiddlewareBundle({
|
||||||
entryPath,
|
entryPath,
|
||||||
outputDirectory,
|
outputDirectory,
|
||||||
@@ -2702,27 +2722,19 @@ export async function getMiddlewareBundle({
|
|||||||
// app/index/page -> index/index
|
// app/index/page -> index/index
|
||||||
if (shortPath.startsWith('pages/')) {
|
if (shortPath.startsWith('pages/')) {
|
||||||
shortPath = shortPath.replace(/^pages\//, '');
|
shortPath = shortPath.replace(/^pages\//, '');
|
||||||
} else if (
|
} else {
|
||||||
shortPath.startsWith('app/') &&
|
shortPath = normalizeEdgeFunctionPath(shortPath, appPathRoutesManifest);
|
||||||
(shortPath.endsWith('/page') ||
|
|
||||||
shortPath.endsWith('/route') ||
|
|
||||||
shortPath === 'app/_not-found')
|
|
||||||
) {
|
|
||||||
const ogRoute = shortPath.replace(/^app\//, '/');
|
|
||||||
shortPath = (
|
|
||||||
appPathRoutesManifest[ogRoute] ||
|
|
||||||
shortPath.replace(/(^|\/)(page|route)$/, '')
|
|
||||||
).replace(/^\//, '');
|
|
||||||
|
|
||||||
if (!shortPath || shortPath === '/') {
|
|
||||||
shortPath = 'index';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (routesManifest?.basePath) {
|
if (routesManifest?.basePath) {
|
||||||
shortPath = path.posix
|
shortPath = normalizeIndexOutput(
|
||||||
.join(routesManifest.basePath, shortPath)
|
path.posix.join(
|
||||||
.replace(/^\//, '');
|
'./',
|
||||||
|
routesManifest?.basePath,
|
||||||
|
shortPath.replace(/^\//, '')
|
||||||
|
),
|
||||||
|
true
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
worker.edgeFunction.name = shortPath;
|
worker.edgeFunction.name = shortPath;
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
|
import Link from 'next/link';
|
||||||
|
|
||||||
export default function Home() {
|
export default function Home() {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<h1>Home</h1>
|
<h1>Home</h1>
|
||||||
|
<Link href="/en/about">About</Link>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
"headers": {
|
"headers": {
|
||||||
"RSC": "1"
|
"RSC": "1"
|
||||||
},
|
},
|
||||||
"mustContain": "<html"
|
"bodyMustBe": "{}"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"path": "/en/foobar",
|
"path": "/en/foobar",
|
||||||
|
|||||||
@@ -2,4 +2,6 @@ export default function handler(req, res) {
|
|||||||
return res.json({ hello: 'world' });
|
return res.json({ hello: 'world' });
|
||||||
}
|
}
|
||||||
|
|
||||||
export const maxDuration = 7;
|
export const config = {
|
||||||
|
maxDuration: 7
|
||||||
|
};
|
||||||
|
|||||||
16
packages/next/test/fixtures/00-app-dir/app/gsp/[text]/page.js
vendored
Normal file
16
packages/next/test/fixtures/00-app-dir/app/gsp/[text]/page.js
vendored
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
export default async function DynamicPage({ params }) {
|
||||||
|
return (
|
||||||
|
<main>
|
||||||
|
<h1>Dynamic page</h1>
|
||||||
|
<p>Param: {params.text}</p>
|
||||||
|
</main>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function generateStaticParams() {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
text: 'one',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
9
packages/next/test/fixtures/00-app-dir/pages/[segmentA]/[segmentB]/[segmentC]/index.js
vendored
Normal file
9
packages/next/test/fixtures/00-app-dir/pages/[segmentA]/[segmentB]/[segmentC]/index.js
vendored
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
export default function Page(props) {
|
||||||
|
return <div>SSRed Page</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getServerSideProps() {
|
||||||
|
return {
|
||||||
|
props: {},
|
||||||
|
}
|
||||||
|
}
|
||||||
3
packages/next/test/fixtures/00-app-dir/pages/docs/categories/foo.js
vendored
Normal file
3
packages/next/test/fixtures/00-app-dir/pages/docs/categories/foo.js
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export default function Page() {
|
||||||
|
return 'Hello World'
|
||||||
|
}
|
||||||
@@ -47,6 +47,15 @@
|
|||||||
"mustContain": ":",
|
"mustContain": ":",
|
||||||
"mustNotContain": "<html"
|
"mustNotContain": "<html"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/categories/foo?_rsc=f7pci",
|
||||||
|
"status": 200,
|
||||||
|
"headers": {
|
||||||
|
"RSC": "1",
|
||||||
|
"Next-Router-Prefetch": "1"
|
||||||
|
},
|
||||||
|
"bodyMustBe": "{}"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"path": "/ssg",
|
"path": "/ssg",
|
||||||
"status": 200,
|
"status": 200,
|
||||||
@@ -235,6 +244,24 @@
|
|||||||
},
|
},
|
||||||
"mustContain": ":{",
|
"mustContain": ":{",
|
||||||
"mustNotContain": "<html"
|
"mustNotContain": "<html"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/gsp/one",
|
||||||
|
"status": 200,
|
||||||
|
"headers": {
|
||||||
|
"RSC": "1"
|
||||||
|
},
|
||||||
|
"mustContain": ":{",
|
||||||
|
"mustNotContain": "<html"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/gsp/two",
|
||||||
|
"status": 200,
|
||||||
|
"headers": {
|
||||||
|
"RSC": "1"
|
||||||
|
},
|
||||||
|
"mustContain": ":{",
|
||||||
|
"mustNotContain": "<html"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
8
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/index.test.js
vendored
Normal file
8
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/index.test.js
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
const path = require('path');
|
||||||
|
const { deployAndTest } = require('../../utils');
|
||||||
|
|
||||||
|
describe(`${__dirname.split(path.sep).pop()}`, () => {
|
||||||
|
it('should deploy and pass probe checks', async () => {
|
||||||
|
await deployAndTest(__dirname);
|
||||||
|
});
|
||||||
|
});
|
||||||
10
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/next.config.js
vendored
Normal file
10
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/next.config.js
vendored
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
module.exports = {
|
||||||
|
generateBuildId() {
|
||||||
|
return 'testing-build-id';
|
||||||
|
},
|
||||||
|
basePath: '/docs',
|
||||||
|
i18n: {
|
||||||
|
locales: ['en', 'fr', 'nl'],
|
||||||
|
defaultLocale: 'en',
|
||||||
|
},
|
||||||
|
};
|
||||||
8
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/package.json
vendored
Normal file
8
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/package.json
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"dependencies": {
|
||||||
|
"next": "canary",
|
||||||
|
"react": "latest",
|
||||||
|
"react-dom": "latest"
|
||||||
|
},
|
||||||
|
"ignoreNextjsUpdates": true
|
||||||
|
}
|
||||||
3
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/404.js
vendored
Normal file
3
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/404.js
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export default function NotFound() {
|
||||||
|
return 'not found page';
|
||||||
|
}
|
||||||
13
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/_app.js
vendored
Normal file
13
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/_app.js
vendored
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
|
||||||
|
function MyApp({ Component, pageProps }) {
|
||||||
|
return <Component {...pageProps} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getStaticProps = ({ locale }) => ({
|
||||||
|
props: {
|
||||||
|
locale
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
export default MyApp;
|
||||||
|
|
||||||
11
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/another.js
vendored
Normal file
11
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/another.js
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
export default function Another() {
|
||||||
|
return 'another page';
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getStaticProps = ({ locale }) => ({
|
||||||
|
props: {
|
||||||
|
locale,
|
||||||
|
hello: 'world',
|
||||||
|
},
|
||||||
|
revalidate: 1,
|
||||||
|
});
|
||||||
3
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/api/blog/[slug].js
vendored
Normal file
3
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/api/blog/[slug].js
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export default function handler(req, res) {
|
||||||
|
res.json({ slug: req.query.slug });
|
||||||
|
}
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
export default function handler(req, res) {
|
||||||
|
res.json({ rest: req.query.rest.join('/') });
|
||||||
|
}
|
||||||
6
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/api/hello.js
vendored
Normal file
6
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/api/hello.js
vendored
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
|
|
||||||
|
export default (req, res) => {
|
||||||
|
res.statusCode = 200;
|
||||||
|
res.json({ name: 'John Doe' });
|
||||||
|
};
|
||||||
25
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/foo/[slug].js
vendored
Normal file
25
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/foo/[slug].js
vendored
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
export default function Page() {
|
||||||
|
return 'Hello World';
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getStaticProps = async ({ locale, params }) => {
|
||||||
|
const id = params?.['slug'];
|
||||||
|
if (!id) {
|
||||||
|
return {
|
||||||
|
notFound: true,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
props: {
|
||||||
|
locale,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getStaticPaths = async () => {
|
||||||
|
return {
|
||||||
|
paths: [{ params: { slug: 'test' } }],
|
||||||
|
fallback: false,
|
||||||
|
};
|
||||||
|
};
|
||||||
3
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/hello.js
vendored
Normal file
3
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/hello.js
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export default function Page() {
|
||||||
|
return 'hello page';
|
||||||
|
}
|
||||||
11
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/index.js
vendored
Normal file
11
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/pages/index.js
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
export default function Home() {
|
||||||
|
return 'index page';
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getServerSideProps = ({ locale }) => ({
|
||||||
|
props: {
|
||||||
|
locale,
|
||||||
|
hello: 'world',
|
||||||
|
gsspData: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
84
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/probes.json
vendored
Normal file
84
packages/next/test/fixtures/00-i18n-basepath-fallback-false-404/probes.json
vendored
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
{
|
||||||
|
"probes": [
|
||||||
|
{
|
||||||
|
"path": "/docs/non-existent",
|
||||||
|
"status": 404,
|
||||||
|
"mustContain": "not found page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/non-existent",
|
||||||
|
"status": 404,
|
||||||
|
"mustContain": "lang=\"en\""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr/non-existent",
|
||||||
|
"status": 404,
|
||||||
|
"mustContain": "not found page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr/non-existent",
|
||||||
|
"status": 404,
|
||||||
|
"mustContain": "lang=\"fr\""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "index page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "index page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/another",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "another page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr/another",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "another page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr/another",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "lang=\"fr\""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/hello",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "hello page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/hello",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "lang=\"en\""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr/hello",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "hello page"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/fr/hello",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "lang=\"fr\""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/api/hello",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "John Doe"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/api/blog/first",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "\"slug\":\"first\""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/docs/api/catchall/hello/world",
|
||||||
|
"status": 200,
|
||||||
|
"mustContain": "\"rest\":\"hello/world\""
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
export const runtime = 'edge';
|
||||||
|
|
||||||
|
const Home = () => {
|
||||||
|
return <div>another</div>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Home;
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
export const runtime = 'edge';
|
||||||
|
|
||||||
|
const Home = () => {
|
||||||
|
return <div>dynamic</div>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Home;
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user