mirror of
https://github.com/LukeHagar/vercel.git
synced 2025-12-12 04:22:14 +00:00
Compare commits
126 Commits
@vercel/py
...
@vercel/py
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a400b9b29d | ||
|
|
b549c37149 | ||
|
|
30d5e64291 | ||
|
|
47c2c361d2 | ||
|
|
438576fc7c | ||
|
|
b30343ef7b | ||
|
|
2dc0dfa572 | ||
|
|
9ee54b3dd6 | ||
|
|
9d67e0bc06 | ||
|
|
466135cf84 | ||
|
|
eab2e229dc | ||
|
|
698b89a2ba | ||
|
|
bae2a2e4df | ||
|
|
57916bb712 | ||
|
|
12bbd4e8eb | ||
|
|
4e4c7023dc | ||
|
|
41805790e7 | ||
|
|
6ad77ae8e1 | ||
|
|
e8daf36cd7 | ||
|
|
c319a2c499 | ||
|
|
a410baa797 | ||
|
|
625568e659 | ||
|
|
41868c1fe0 | ||
|
|
1b644f1218 | ||
|
|
29ea0fb06b | ||
|
|
b61f049f11 | ||
|
|
16e28f326b | ||
|
|
276397c940 | ||
|
|
85d7311199 | ||
|
|
b8114b8b39 | ||
|
|
d63e8d3187 | ||
|
|
11d3dd04aa | ||
|
|
46bf95ee36 | ||
|
|
92252468c2 | ||
|
|
71b83d5587 | ||
|
|
d9e5fdc5e4 | ||
|
|
58f479c603 | ||
|
|
d62461d952 | ||
|
|
e7f524defb | ||
|
|
bdefd0d05d | ||
|
|
ca522fc9f1 | ||
|
|
4a8504fc45 | ||
|
|
576217b344 | ||
|
|
f03129ac7a | ||
|
|
3eaad1fd52 | ||
|
|
4e471491d8 | ||
|
|
99395fd9fe | ||
|
|
4980fe6513 | ||
|
|
24676ae020 | ||
|
|
72ada9abd8 | ||
|
|
da893e7c57 | ||
|
|
e40eecafc9 | ||
|
|
d9e5342eba | ||
|
|
b0ab13778d | ||
|
|
c76dfbe8c9 | ||
|
|
ed6f3cc27e | ||
|
|
7b0186bffe | ||
|
|
45e22b6b60 | ||
|
|
b2bfae6e2e | ||
|
|
b6ed718f52 | ||
|
|
a84c4a37e6 | ||
|
|
d2ba06c6cb | ||
|
|
06e7753df0 | ||
|
|
235a9c3300 | ||
|
|
1b70402325 | ||
|
|
c76781fac9 | ||
|
|
451e0b0cfb | ||
|
|
cf477d45b2 | ||
|
|
cdd2d69e07 | ||
|
|
43f1f8b257 | ||
|
|
34055e3599 | ||
|
|
26abb0a85a | ||
|
|
a1e337e0dd | ||
|
|
b72ead480f | ||
|
|
77cf68105f | ||
|
|
d800f55dfa | ||
|
|
9dde99f19e | ||
|
|
fae7a083fc | ||
|
|
cbd651d6ee | ||
|
|
6077a706d1 | ||
|
|
cedc82dd9e | ||
|
|
b420006401 | ||
|
|
8ba604e8fc | ||
|
|
fadeee4568 | ||
|
|
2b15ba7f46 | ||
|
|
4cdfd0e58c | ||
|
|
b3ccb5f3ef | ||
|
|
584acc43b7 | ||
|
|
f64be93b94 | ||
|
|
9abd31769e | ||
|
|
09e3b35e74 | ||
|
|
8aa9a0ea05 | ||
|
|
b2d0ed74c6 | ||
|
|
aef936af0f | ||
|
|
501be936c0 | ||
|
|
7eba282af5 | ||
|
|
cf3e4bd726 | ||
|
|
ee5361b00e | ||
|
|
bd929dd5c5 | ||
|
|
4ee064bb61 | ||
|
|
312a2090a6 | ||
|
|
a82f117217 | ||
|
|
e908378486 | ||
|
|
8cda5263eb | ||
|
|
a24fd64bce | ||
|
|
0c515a46d5 | ||
|
|
f19690dc32 | ||
|
|
6b2a1c3866 | ||
|
|
1e54d606d7 | ||
|
|
c4ab0ebe9c | ||
|
|
321f1232a1 | ||
|
|
8a8203e149 | ||
|
|
33527165e7 | ||
|
|
db10383bc8 | ||
|
|
56960e506e | ||
|
|
a17f3a96ec | ||
|
|
9ff86a896c | ||
|
|
6ccb4354f9 | ||
|
|
82ba932447 | ||
|
|
b995618afb | ||
|
|
c897b24417 | ||
|
|
e64b2e1993 | ||
|
|
9358a5469e | ||
|
|
2a24210b7d | ||
|
|
2644a59984 | ||
|
|
4eeb8c298c |
@@ -1,7 +1,12 @@
|
|||||||
node_modules
|
node_modules
|
||||||
dist
|
dist
|
||||||
examples
|
examples
|
||||||
|
|
||||||
|
packages/node/src/bridge.ts
|
||||||
|
|
||||||
packages/*/test/fixtures
|
packages/*/test/fixtures
|
||||||
|
|
||||||
|
# cli
|
||||||
packages/cli/@types
|
packages/cli/@types
|
||||||
packages/cli/download
|
packages/cli/download
|
||||||
packages/cli/dist
|
packages/cli/dist
|
||||||
@@ -9,11 +14,24 @@ packages/cli/test/dev/fixtures
|
|||||||
packages/cli/bin
|
packages/cli/bin
|
||||||
packages/cli/link
|
packages/cli/link
|
||||||
packages/cli/src/util/dev/templates/*.ts
|
packages/cli/src/util/dev/templates/*.ts
|
||||||
|
|
||||||
|
# client
|
||||||
packages/client/tests/fixtures
|
packages/client/tests/fixtures
|
||||||
packages/client/lib
|
packages/client/lib
|
||||||
packages/node/src/bridge.ts
|
|
||||||
|
# node-bridge
|
||||||
packages/node-bridge/bridge.js
|
packages/node-bridge/bridge.js
|
||||||
packages/node-bridge/launcher.js
|
packages/node-bridge/launcher.js
|
||||||
packages/node-bridge/helpers.js
|
packages/node-bridge/helpers.js
|
||||||
packages/node-bridge/source-map-support.js
|
packages/node-bridge/source-map-support.js
|
||||||
|
|
||||||
|
# middleware
|
||||||
packages/middleware/src/entries.js
|
packages/middleware/src/entries.js
|
||||||
|
|
||||||
|
# static-build
|
||||||
|
packages/static-build/test/fixtures
|
||||||
|
packages/static-build/test/build-fixtures
|
||||||
|
packages/static-build/test/cache-fixtures
|
||||||
|
|
||||||
|
# redwood
|
||||||
|
packages/redwood/test/fixtures
|
||||||
|
|||||||
6
.github/CONTRIBUTING.md
vendored
6
.github/CONTRIBUTING.md
vendored
@@ -16,17 +16,17 @@ yarn install
|
|||||||
yarn bootstrap
|
yarn bootstrap
|
||||||
yarn build
|
yarn build
|
||||||
yarn lint
|
yarn lint
|
||||||
yarn test
|
yarn test-unit
|
||||||
```
|
```
|
||||||
|
|
||||||
Make sure all the tests pass before making changes.
|
Make sure all the tests pass before making changes.
|
||||||
|
|
||||||
## Verifying your change
|
## Verifying your change
|
||||||
|
|
||||||
Once you are done with your changes (we even suggest doing it along the way), make sure all the test still run by running:
|
Once you are done with your changes (we even suggest doing it along the way), make sure all the test still pass by running:
|
||||||
|
|
||||||
```
|
```
|
||||||
yarn build && yarn test
|
yarn test-unit
|
||||||
```
|
```
|
||||||
|
|
||||||
from the root of the project.
|
from the root of the project.
|
||||||
|
|||||||
2
.github/workflows/test-integration-cli.yml
vendored
2
.github/workflows/test-integration-cli.yml
vendored
@@ -11,7 +11,7 @@ on:
|
|||||||
jobs:
|
jobs:
|
||||||
test:
|
test:
|
||||||
name: CLI
|
name: CLI
|
||||||
timeout-minutes: 30
|
timeout-minutes: 40
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
!.yarnrc
|
!.yarnrc
|
||||||
!yarn.lock
|
!yarn.lock
|
||||||
!package.json
|
!package.json
|
||||||
|
!turbo.json
|
||||||
|
|
||||||
# api
|
# api
|
||||||
!api/
|
!api/
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ A Runtime is an npm module that implements the following interface:
|
|||||||
interface Runtime {
|
interface Runtime {
|
||||||
version: number;
|
version: number;
|
||||||
build: (options: BuildOptions) => Promise<BuildResult>;
|
build: (options: BuildOptions) => Promise<BuildResult>;
|
||||||
analyze?: (options: AnalyzeOptions) => Promise<string>;
|
|
||||||
prepareCache?: (options: PrepareCacheOptions) => Promise<CacheOutputs>;
|
prepareCache?: (options: PrepareCacheOptions) => Promise<CacheOutputs>;
|
||||||
shouldServe?: (options: ShouldServeOptions) => Promise<boolean>;
|
shouldServe?: (options: ShouldServeOptions) => Promise<boolean>;
|
||||||
startDevServer?: (
|
startDevServer?: (
|
||||||
@@ -72,26 +71,6 @@ export async function build(options: BuildOptions) {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
### `analyze()`
|
|
||||||
|
|
||||||
An **optional** exported function that returns a unique fingerprint used for the
|
|
||||||
purpose of [build
|
|
||||||
de-duplication](https://vercel.com/docs/v2/platform/deployments#deduplication).
|
|
||||||
If the `analyze()` function is not supplied, then a random fingerprint is
|
|
||||||
assigned to each build.
|
|
||||||
|
|
||||||
**Example:**
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
import { AnalyzeOptions } from '@vercel/build-utils';
|
|
||||||
|
|
||||||
export async function analyze(options: AnalyzeOptions) {
|
|
||||||
// Do calculations to generate a fingerprint based off the source code here…
|
|
||||||
|
|
||||||
return 'fingerprint goes here';
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### `prepareCache()`
|
### `prepareCache()`
|
||||||
|
|
||||||
An **optional** exported function that is executed after [`build()`](#build) is
|
An **optional** exported function that is executed after [`build()`](#build) is
|
||||||
@@ -328,15 +307,15 @@ This is a [class](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Refere
|
|||||||
|
|
||||||
This is an abstract enumeration type that is implemented by one of the following possible `String` values:
|
This is an abstract enumeration type that is implemented by one of the following possible `String` values:
|
||||||
|
|
||||||
|
- `nodejs14.x`
|
||||||
- `nodejs12.x`
|
- `nodejs12.x`
|
||||||
- `nodejs10.x`
|
|
||||||
- `go1.x`
|
- `go1.x`
|
||||||
- `java11`
|
- `java11`
|
||||||
- `python3.9`
|
- `python3.9`
|
||||||
- `python3.6`
|
- `dotnet6`
|
||||||
- `dotnetcore2.1`
|
- `dotnetcore3.1`
|
||||||
- `ruby2.5`
|
- `ruby2.7`
|
||||||
- `provided`
|
- `provided.al2`
|
||||||
|
|
||||||
## `@vercel/build-utils` Helper Functions
|
## `@vercel/build-utils` Helper Functions
|
||||||
|
|
||||||
|
|||||||
46
api/_lib/script/build.ts
Normal file
46
api/_lib/script/build.ts
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import fs from 'fs/promises';
|
||||||
|
import { join } from 'path';
|
||||||
|
import { getExampleList } from '../examples/example-list';
|
||||||
|
import { mapOldToNew } from '../examples/map-old-to-new';
|
||||||
|
|
||||||
|
const repoRoot = join(__dirname, '..', '..', '..');
|
||||||
|
const pubDir = join(repoRoot, 'public');
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
console.log(`Building static frontend ${repoRoot}...`);
|
||||||
|
|
||||||
|
await fs.rm(pubDir, { recursive: true, force: true });
|
||||||
|
await fs.mkdir(pubDir);
|
||||||
|
|
||||||
|
const examples = await getExampleList();
|
||||||
|
const pathListAll = join(pubDir, 'list-all.json');
|
||||||
|
await fs.writeFile(pathListAll, JSON.stringify(examples));
|
||||||
|
|
||||||
|
const exampleDirs = await fs.readdir(join(repoRoot, 'examples'), {
|
||||||
|
withFileTypes: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
const existingExamples = exampleDirs
|
||||||
|
.filter(dir => dir.isDirectory())
|
||||||
|
.map(dir => ({
|
||||||
|
name: dir.name,
|
||||||
|
visible: true,
|
||||||
|
suggestions: [],
|
||||||
|
}));
|
||||||
|
|
||||||
|
const oldExamples = Object.keys(mapOldToNew).map(key => ({
|
||||||
|
name: key,
|
||||||
|
visible: false,
|
||||||
|
suggestions: mapOldToNew[key],
|
||||||
|
}));
|
||||||
|
|
||||||
|
const pathList = join(pubDir, 'list.json');
|
||||||
|
await fs.writeFile(
|
||||||
|
pathList,
|
||||||
|
JSON.stringify([...existingExamples, ...oldExamples])
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log('Completed building static frontend.');
|
||||||
|
}
|
||||||
|
|
||||||
|
main().catch(console.error);
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
import { VercelRequest, VercelResponse } from '@vercel/node';
|
|
||||||
import { getExampleList } from '../_lib/examples/example-list';
|
|
||||||
import { withApiHandler } from '../_lib/util/with-api-handler';
|
|
||||||
|
|
||||||
export default withApiHandler(async function (
|
|
||||||
req: VercelRequest,
|
|
||||||
res: VercelResponse
|
|
||||||
) {
|
|
||||||
res.status(200).json(await getExampleList());
|
|
||||||
});
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
import { extract } from '../_lib/examples/extract';
|
|
||||||
import { summary } from '../_lib/examples/summary';
|
|
||||||
import { VercelRequest, VercelResponse } from '@vercel/node';
|
|
||||||
import { mapOldToNew } from '../_lib/examples/map-old-to-new';
|
|
||||||
import { withApiHandler } from '../_lib/util/with-api-handler';
|
|
||||||
|
|
||||||
export default withApiHandler(async function (
|
|
||||||
req: VercelRequest,
|
|
||||||
res: VercelResponse
|
|
||||||
) {
|
|
||||||
await extract('https://github.com/vercel/vercel/archive/main.zip', '/tmp');
|
|
||||||
const exampleList = summary('/tmp/vercel-main/examples');
|
|
||||||
|
|
||||||
const existingExamples = Array.from(exampleList).map(key => ({
|
|
||||||
name: key,
|
|
||||||
visible: true,
|
|
||||||
suggestions: [],
|
|
||||||
}));
|
|
||||||
|
|
||||||
const oldExamples = Object.keys(mapOldToNew).map(key => ({
|
|
||||||
name: key,
|
|
||||||
visible: false,
|
|
||||||
suggestions: mapOldToNew[key],
|
|
||||||
}));
|
|
||||||
|
|
||||||
res.status(200).json([...existingExamples, ...oldExamples]);
|
|
||||||
});
|
|
||||||
@@ -16,4 +16,4 @@ If you would not like to verify your domain, you can remove it from your account
|
|||||||
|
|
||||||
#### Resources
|
#### Resources
|
||||||
|
|
||||||
- [Vercel Custom Domains Documentation](https://vercel.com/docs/v2/custom-domains)
|
- [Vercel Custom Domains Documentation](https://vercel.com/docs/concepts/projects/custom-domains)
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
#### Why This Error Occurred
|
#### Why This Error Occurred
|
||||||
|
|
||||||
You ran `vercel dev` inside a project that contains a `vercel.json` file with `env` or `build.env` properties that use [Vercel Secrets](https://vercel.com/docs/v2/build-step#environment-variables).
|
You ran `vercel dev` inside a project that contains a `vercel.json` file with `env` or `build.env` properties that use [Vercel Secrets](https://vercel.com/docs/concepts/projects/environment-variables).
|
||||||
|
|
||||||
In order to use environment variables in your project locally that have values defined using the Vercel Secrets format (e.g. `@my-secret-value`), you will need to provide the value as an environment variable using a `.env`.
|
In order to use environment variables in your project locally that have values defined using the Vercel Secrets format (e.g. `@my-secret-value`), you will need to provide the value as an environment variable using a `.env`.
|
||||||
|
|
||||||
@@ -24,4 +24,4 @@ TEST=value
|
|||||||
|
|
||||||
In the above example, `TEST` represents the name of the environment variable and `value` its value.
|
In the above example, `TEST` represents the name of the environment variable and `value` its value.
|
||||||
|
|
||||||
For more information on Environment Variables in development, [see the documentation](https://vercel.com/docs/v2/build-step#environment-variables).
|
For more information on Environment Variables in development, [see the documentation](https://vercel.com/docs/concepts/projects/environment-variables).
|
||||||
|
|||||||
1
examples/amp/.gitignore
vendored
1
examples/amp/.gitignore
vendored
@@ -1 +0,0 @@
|
|||||||
.env
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
# AMP Example
|
|
||||||
|
|
||||||
This directory is a brief example of an [AMP](https://amp.dev/) site that can be deployed to Vercel with zero configuration.
|
|
||||||
|
|
||||||
## Deploy Your Own
|
|
||||||
|
|
||||||
Deploy your own AMP project with Vercel.
|
|
||||||
|
|
||||||
[](https://vercel.com/new/clone?repository-url=https://github.com/vercel/vercel/tree/main/examples/amp)
|
|
||||||
|
|
||||||
_Live Example: https://amp-template.vercel.app_
|
|
||||||
|
|
||||||
### How We Created This Example
|
|
||||||
|
|
||||||
To get started deploying AMP with Vercel, you can use the [Vercel CLI](https://vercel.com/download) to initialize the project:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
$ vercel init amp
|
|
||||||
```
|
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 2.0 KiB |
@@ -1,72 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html ⚡>
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<meta name="viewport" content="width=device-width,minimum-scale=1" />
|
|
||||||
<link rel="shortcut icon" href="favicon.png">
|
|
||||||
<style amp-boilerplate>body{-webkit-animation:-amp-start 8s steps(1,end) 0s 1 normal both;-moz-animation:-amp-start 8s steps(1,end) 0s 1 normal both;-ms-animation:-amp-start 8s steps(1,end) 0s 1 normal both;animation:-amp-start 8s steps(1,end) 0s 1 normal both}@-webkit-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@-moz-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@-ms-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@-o-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}</style><noscript><style amp-boilerplate>body{-webkit-animation:none;-moz-animation:none;-ms-animation:none;animation:none}</style></noscript>
|
|
||||||
<link rel="canonical" href="index.html" />
|
|
||||||
<title>AMP Website</title>
|
|
||||||
<script async src="https://cdn.ampproject.org/v0.js"></script>
|
|
||||||
<style amp-custom>
|
|
||||||
body > * {
|
|
||||||
margin: 3rem 1rem;
|
|
||||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
|
|
||||||
color: #525252;
|
|
||||||
}
|
|
||||||
|
|
||||||
h3 {
|
|
||||||
font-size: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
h4 {
|
|
||||||
margin-top: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
p {
|
|
||||||
font-size: 1.2rem;
|
|
||||||
line-height: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.links {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
margin-bottom: 3rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.links a {
|
|
||||||
margin: 0 10px;
|
|
||||||
font-size: 1rem;
|
|
||||||
color: #005af0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<center>
|
|
||||||
<amp-img width=150 height=150 layout="fixed" class="logo" src="logo.png"></amp-img>
|
|
||||||
<h3>Welcome to your AMP page</h3>
|
|
||||||
<p>AMP is a web component framework to <br> easily create user-first websites, stories, ads and emails.</p>
|
|
||||||
|
|
||||||
<h4>Links</h4>
|
|
||||||
<div class="links">
|
|
||||||
<a href="https://amp.dev/">Homepage</a>
|
|
||||||
<a href="https://amp.dev/documentation/guides-and-tutorials/?format=websites">Tutorials</a>
|
|
||||||
<a href="https://amp.dev/documentation/examples/">Examples</a>
|
|
||||||
<a href="https://blog.amp.dev">Blog</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4>Ready to get started?</h4>
|
|
||||||
<div class="links">
|
|
||||||
<a href="https://amp.dev/documentation/guides-and-tutorials/start/create/?format=websites">Create your first AMP page</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4>Get involved</h4>
|
|
||||||
<div class="links">
|
|
||||||
<a href="https://twitter.com/amphtml">Twitter</a>
|
|
||||||
<a href="https://amphtml.slack.com">Slack</a>
|
|
||||||
<a href="https://amp.dev/events/amp-conf-2019">AMP Conf</a>
|
|
||||||
<a href="https://amp.dev/events/amp-roadshow">AMP Roadshow</a>
|
|
||||||
</div>
|
|
||||||
</center>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 43 KiB |
@@ -10,7 +10,7 @@ Deploy your own Angular project with Vercel.
|
|||||||
|
|
||||||
[](https://vercel.com/new/clone?repository-url=https://github.com/vercel/vercel/tree/main/examples/angular&template=angular)
|
[](https://vercel.com/new/clone?repository-url=https://github.com/vercel/vercel/tree/main/examples/angular&template=angular)
|
||||||
|
|
||||||
_Live Example: https://angular-now-examples.vercel.app_
|
_Live Example: https://angular-template.vercel.app_
|
||||||
|
|
||||||
## Development server
|
## Development server
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ This directory is a brief example of a [Dojo](https://dojo.io) site that can be
|
|||||||
|
|
||||||
Deploy your own Dojo project with Vercel.
|
Deploy your own Dojo project with Vercel.
|
||||||
|
|
||||||
[](https://vercel.com/new/clone?repository-url=https://github.com/vercel/vercel/tree/main/dojo&template=dojo)
|
[](https://vercel.com/new/clone?repository-url=https://github.com/vercel/vercel/tree/main/examples/dojo&template=dojo)
|
||||||
|
|
||||||
### How We Created This Example
|
### How We Created This Example
|
||||||
|
|
||||||
|
|||||||
20
examples/eleventy/LICENSE.md
Normal file
20
examples/eleventy/LICENSE.md
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
MIT License
|
||||||
|
Copyright (c) 2022 Zach Leatherman @zachleat
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
@@ -1,5 +1,19 @@
|
|||||||
{
|
{
|
||||||
"private": true,
|
"private": true,
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "git://github.com/11ty/eleventy-base-blog.git"
|
||||||
|
},
|
||||||
|
"author": {
|
||||||
|
"name": "Zach Leatherman",
|
||||||
|
"email": "zachleatherman@gmail.com",
|
||||||
|
"url": "https://zachleat.com/"
|
||||||
|
},
|
||||||
|
"license": "MIT",
|
||||||
|
"bugs": {
|
||||||
|
"url": "https://github.com/11ty/eleventy-base-blog/issues"
|
||||||
|
},
|
||||||
|
"homepage": "https://github.com/11ty/eleventy-base-blog#readme",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "eleventy",
|
"build": "eleventy",
|
||||||
"watch": "eleventy --watch",
|
"watch": "eleventy --watch",
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ cache:
|
|||||||
|
|
||||||
env:
|
env:
|
||||||
global:
|
global:
|
||||||
# See https://git.io/vdao3 for details.
|
# See https://github.com/ember-cli/ember-cli/blob/master/docs/build-concurrency.md
|
||||||
- JOBS=1
|
- JOBS=1
|
||||||
|
|
||||||
script:
|
script:
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ async function main() {
|
|||||||
// if it's an empty string, just ignore it
|
// if it's an empty string, just ignore it
|
||||||
if (!formatted) return false;
|
if (!formatted) return false;
|
||||||
|
|
||||||
let type = url.substr(-3) == 'css' ? 'style' : 'script';
|
let type = url.slice(-3) == 'css' ? 'style' : 'script';
|
||||||
results += `</${formatted}>;rel=preload;as=${type},`;
|
results += `</${formatted}>;rel=preload;as=${type},`;
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|||||||
4882
examples/nextjs/package-lock.json
generated
Normal file
4882
examples/nextjs/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -7,12 +7,12 @@
|
|||||||
"lint": "next lint"
|
"lint": "next lint"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"next": "12.1.0",
|
"next": "12.1.4",
|
||||||
"react": "17.0.2",
|
"react": "18.0.0",
|
||||||
"react-dom": "17.0.2"
|
"react-dom": "18.0.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"eslint": "8.9.0",
|
"eslint": "8.12.0",
|
||||||
"eslint-config-next": "12.1.0"
|
"eslint-config-next": "12.1.4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -31,7 +31,7 @@ export default function Index() {
|
|||||||
<p>
|
<p>
|
||||||
Wait a sec...<em>its children</em>? To understand what we mean by
|
Wait a sec...<em>its children</em>? To understand what we mean by
|
||||||
this,{" "}
|
this,{" "}
|
||||||
<a href="https://remix.run/tutorial/4-nested-routes-params">
|
<a href="https://remix.run/docs/en/v1/guides/routing">
|
||||||
read all about nested routes in the docs
|
read all about nested routes in the docs
|
||||||
</a>
|
</a>
|
||||||
.
|
.
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ export class AppProfile {
|
|||||||
|
|
||||||
normalize(name: string): string {
|
normalize(name: string): string {
|
||||||
if (name) {
|
if (name) {
|
||||||
return name.substr(0, 1).toUpperCase() + name.substr(1).toLowerCase();
|
return name.slice(0, 1).toUpperCase() + name.slice(1).toLowerCase();
|
||||||
}
|
}
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|||||||
670
examples/sveltekit/package-lock.json
generated
670
examples/sveltekit/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,7 @@
|
|||||||
{
|
{
|
||||||
"private": true,
|
"private": true,
|
||||||
|
"name": "sveltekit",
|
||||||
|
"version": "0.0.1",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "svelte-kit dev",
|
"dev": "svelte-kit dev",
|
||||||
"build": "svelte-kit build",
|
"build": "svelte-kit build",
|
||||||
@@ -9,7 +11,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@sveltejs/adapter-auto": "next",
|
"@sveltejs/adapter-auto": "next",
|
||||||
"@sveltejs/kit": "next",
|
"@sveltejs/kit": "next",
|
||||||
"svelte": "^3.44.0"
|
"svelte": "^3.46.0"
|
||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|||||||
@@ -8,6 +8,6 @@
|
|||||||
%svelte.head%
|
%svelte.head%
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="svelte">%svelte.body%</div>
|
<div>%svelte.body%</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
1
examples/sveltekit/src/global.d.ts
vendored
1
examples/sveltekit/src/global.d.ts
vendored
@@ -1 +0,0 @@
|
|||||||
/// <reference types="@sveltejs/kit" />
|
|
||||||
@@ -1,19 +1,22 @@
|
|||||||
import cookie from 'cookie';
|
import cookie from 'cookie';
|
||||||
import { v4 as uuid } from '@lukeed/uuid';
|
import { v4 as uuid } from '@lukeed/uuid';
|
||||||
|
|
||||||
export const handle = async ({ request, resolve }) => {
|
export const handle = async ({ event, resolve }) => {
|
||||||
const cookies = cookie.parse(request.headers.cookie || '');
|
const cookies = cookie.parse(event.request.headers.get('cookie') || '');
|
||||||
request.locals.userid = cookies.userid || uuid();
|
event.locals.userid = cookies.userid || uuid();
|
||||||
|
|
||||||
const response = await resolve(request);
|
const response = await resolve(event);
|
||||||
|
|
||||||
if (!cookies.userid) {
|
if (!cookies.userid) {
|
||||||
// if this is the first time the user has visited this app,
|
// if this is the first time the user has visited this app,
|
||||||
// set a cookie so that we recognise them when they return
|
// set a cookie so that we recognise them when they return
|
||||||
response.headers['set-cookie'] = cookie.serialize('userid', request.locals.userid, {
|
response.headers.set(
|
||||||
|
'set-cookie',
|
||||||
|
cookie.serialize('userid', event.locals.userid, {
|
||||||
path: '/',
|
path: '/',
|
||||||
httpOnly: true
|
httpOnly: true
|
||||||
});
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
|
|||||||
@@ -22,7 +22,7 @@
|
|||||||
|
|
||||||
<div class="counter-viewport">
|
<div class="counter-viewport">
|
||||||
<div class="counter-digits" style="transform: translate(0, {100 * offset}%)">
|
<div class="counter-digits" style="transform: translate(0, {100 * offset}%)">
|
||||||
<strong style="top: -100%" aria-hidden="true">{Math.floor($displayed_count + 1)}</strong>
|
<strong class="hidden" aria-hidden="true">{Math.floor($displayed_count + 1)}</strong>
|
||||||
<strong>{Math.floor($displayed_count)}</strong>
|
<strong>{Math.floor($displayed_count)}</strong>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -94,4 +94,9 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.hidden {
|
||||||
|
top: -100%;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
import { invalidate } from '$app/navigation';
|
||||||
|
|
||||||
// this action (https://svelte.dev/tutorial/actions) allows us to
|
// this action (https://svelte.dev/tutorial/actions) allows us to
|
||||||
// progressively enhance a <form> that already works without JS
|
// progressively enhance a <form> that already works without JS
|
||||||
export function enhance(form, { pending, error, result }) {
|
export function enhance(form, { pending, error, result } = {}) {
|
||||||
let current_token;
|
let current_token;
|
||||||
|
|
||||||
async function handle_submit(e) {
|
async function handle_submit(e) {
|
||||||
@@ -8,31 +10,35 @@ export function enhance(form, { pending, error, result }) {
|
|||||||
|
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
const body = new FormData(form);
|
const data = new FormData(form);
|
||||||
|
|
||||||
if (pending) pending(body, form);
|
if (pending) pending({ data, form });
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const res = await fetch(form.action, {
|
const response = await fetch(form.action, {
|
||||||
method: form.method,
|
method: form.method,
|
||||||
headers: {
|
headers: {
|
||||||
accept: 'application/json'
|
accept: 'application/json'
|
||||||
},
|
},
|
||||||
body
|
body: data
|
||||||
});
|
});
|
||||||
|
|
||||||
if (token !== current_token) return;
|
if (token !== current_token) return;
|
||||||
|
|
||||||
if (res.ok) {
|
if (response.ok) {
|
||||||
result(res, form);
|
if (result) result({ data, form, response });
|
||||||
|
|
||||||
|
const url = new URL(form.action);
|
||||||
|
url.search = url.hash = '';
|
||||||
|
invalidate(url.href);
|
||||||
} else if (error) {
|
} else if (error) {
|
||||||
error(res, null, form);
|
error({ data, form, error: null, response });
|
||||||
} else {
|
} else {
|
||||||
console.error(await res.text());
|
console.error(await response.text());
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (error) {
|
if (error) {
|
||||||
error(null, e, form);
|
error({ data, form, error: e, response: null });
|
||||||
} else {
|
} else {
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +0,0 @@
|
|||||||
import { api } from './_api';
|
|
||||||
|
|
||||||
// PATCH /todos/:uid.json
|
|
||||||
export const patch = async (request) => {
|
|
||||||
return api(request, `todos/${request.locals.userid}/${request.params.uid}`, {
|
|
||||||
text: request.body.get('text'),
|
|
||||||
done: request.body.has('done') ? !!request.body.get('done') : undefined
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
// DELETE /todos/:uid.json
|
|
||||||
export const del = async (request) => {
|
|
||||||
return api(request, `todos/${request.locals.userid}/${request.params.uid}`);
|
|
||||||
};
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
This module is used by the /todos.json and /todos/[uid].json
|
This module is used by the /todos endpoint to
|
||||||
endpoints to make calls to api.svelte.dev, which stores todos
|
make calls to api.svelte.dev, which stores todos
|
||||||
for each user. The leading underscore indicates that this is
|
for each user. The leading underscore indicates that this is
|
||||||
a private module, _not_ an endpoint — visiting /todos/_api
|
a private module, _not_ an endpoint — visiting /todos/_api
|
||||||
will net you a 404 response.
|
will net you a 404 response.
|
||||||
@@ -11,35 +11,12 @@
|
|||||||
|
|
||||||
const base = 'https://api.svelte.dev';
|
const base = 'https://api.svelte.dev';
|
||||||
|
|
||||||
export async function api(request, resource, data) {
|
export function api(method, resource, data) {
|
||||||
// user must have a cookie set
|
return fetch(`${base}/${resource}`, {
|
||||||
if (!request.locals.userid) {
|
method,
|
||||||
return { status: 401 };
|
|
||||||
}
|
|
||||||
|
|
||||||
const res = await fetch(`${base}/${resource}`, {
|
|
||||||
method: request.method,
|
|
||||||
headers: {
|
headers: {
|
||||||
'content-type': 'application/json'
|
'content-type': 'application/json'
|
||||||
},
|
},
|
||||||
body: data && JSON.stringify(data)
|
body: data && JSON.stringify(data)
|
||||||
});
|
});
|
||||||
|
|
||||||
// if the request came from a <form> submission, the browser's default
|
|
||||||
// behaviour is to show the URL corresponding to the form's "action"
|
|
||||||
// attribute. in those cases, we want to redirect them back to the
|
|
||||||
// /todos page, rather than showing the response
|
|
||||||
if (res.ok && request.method !== 'GET' && request.headers.accept !== 'application/json') {
|
|
||||||
return {
|
|
||||||
status: 303,
|
|
||||||
headers: {
|
|
||||||
location: '/todos'
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
status: res.status,
|
|
||||||
body: await res.json()
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|||||||
66
examples/sveltekit/src/routes/todos/index.js
Normal file
66
examples/sveltekit/src/routes/todos/index.js
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
import { api } from './_api';
|
||||||
|
|
||||||
|
export const get = async ({ locals }) => {
|
||||||
|
// locals.userid comes from src/hooks.js
|
||||||
|
const response = await api('get', `todos/${locals.userid}`);
|
||||||
|
|
||||||
|
if (response.status === 404) {
|
||||||
|
// user hasn't created a todo list.
|
||||||
|
// start with an empty array
|
||||||
|
return {
|
||||||
|
body: {
|
||||||
|
todos: []
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status === 200) {
|
||||||
|
return {
|
||||||
|
body: {
|
||||||
|
todos: await response.json()
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: response.status
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export const post = async ({ request, locals }) => {
|
||||||
|
const form = await request.formData();
|
||||||
|
|
||||||
|
await api('post', `todos/${locals.userid}`, {
|
||||||
|
text: form.get('text')
|
||||||
|
});
|
||||||
|
|
||||||
|
return {};
|
||||||
|
};
|
||||||
|
|
||||||
|
// If the user has JavaScript disabled, the URL will change to
|
||||||
|
// include the method override unless we redirect back to /todos
|
||||||
|
const redirect = {
|
||||||
|
status: 303,
|
||||||
|
headers: {
|
||||||
|
location: '/todos'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const patch = async ({ request, locals }) => {
|
||||||
|
const form = await request.formData();
|
||||||
|
|
||||||
|
await api('patch', `todos/${locals.userid}/${form.get('uid')}`, {
|
||||||
|
text: form.has('text') ? form.get('text') : undefined,
|
||||||
|
done: form.has('done') ? !!form.get('done') : undefined
|
||||||
|
});
|
||||||
|
|
||||||
|
return redirect;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const del = async ({ request, locals }) => {
|
||||||
|
const form = await request.formData();
|
||||||
|
|
||||||
|
await api('delete', `todos/${locals.userid}/${form.get('uid')}`);
|
||||||
|
|
||||||
|
return redirect;
|
||||||
|
};
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
import { api } from './_api';
|
|
||||||
|
|
||||||
// GET /todos.json
|
|
||||||
export const get = async (request) => {
|
|
||||||
// request.locals.userid comes from src/hooks.js
|
|
||||||
const response = await api(request, `todos/${request.locals.userid}`);
|
|
||||||
|
|
||||||
if (response.status === 404) {
|
|
||||||
// user hasn't created a todo list.
|
|
||||||
// start with an empty array
|
|
||||||
return { body: [] };
|
|
||||||
}
|
|
||||||
|
|
||||||
return response;
|
|
||||||
};
|
|
||||||
|
|
||||||
// POST /todos.json
|
|
||||||
export const post = async (request) => {
|
|
||||||
const response = await api(request, `todos/${request.locals.userid}`, {
|
|
||||||
// because index.svelte posts a FormData object,
|
|
||||||
// request.body is _also_ a (readonly) FormData
|
|
||||||
// object, which allows us to get form data
|
|
||||||
// with the `body.get(key)` method
|
|
||||||
text: request.body.get('text')
|
|
||||||
});
|
|
||||||
|
|
||||||
return response;
|
|
||||||
};
|
|
||||||
@@ -1,40 +1,9 @@
|
|||||||
<script context="module">
|
|
||||||
import { enhance } from '$lib/form';
|
|
||||||
|
|
||||||
// see https://kit.svelte.dev/docs#loading
|
|
||||||
export const load = async ({ fetch }) => {
|
|
||||||
const res = await fetch('/todos.json');
|
|
||||||
|
|
||||||
if (res.ok) {
|
|
||||||
const todos = await res.json();
|
|
||||||
|
|
||||||
return {
|
|
||||||
props: { todos }
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const { message } = await res.json();
|
|
||||||
|
|
||||||
return {
|
|
||||||
error: new Error(message)
|
|
||||||
};
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import { enhance } from '$lib/form';
|
||||||
import { scale } from 'svelte/transition';
|
import { scale } from 'svelte/transition';
|
||||||
import { flip } from 'svelte/animate';
|
import { flip } from 'svelte/animate';
|
||||||
|
|
||||||
export let todos;
|
export let todos;
|
||||||
|
|
||||||
async function patch(res) {
|
|
||||||
const todo = await res.json();
|
|
||||||
|
|
||||||
todos = todos.map((t) => {
|
|
||||||
if (t.uid === todo.uid) return todo;
|
|
||||||
return t;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<svelte:head>
|
<svelte:head>
|
||||||
@@ -46,13 +15,10 @@
|
|||||||
|
|
||||||
<form
|
<form
|
||||||
class="new"
|
class="new"
|
||||||
action="/todos.json"
|
action="/todos"
|
||||||
method="post"
|
method="post"
|
||||||
use:enhance={{
|
use:enhance={{
|
||||||
result: async (res, form) => {
|
result: async ({ form }) => {
|
||||||
const created = await res.json();
|
|
||||||
todos = [...todos, created];
|
|
||||||
|
|
||||||
form.reset();
|
form.reset();
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
@@ -68,41 +34,33 @@
|
|||||||
animate:flip={{ duration: 200 }}
|
animate:flip={{ duration: 200 }}
|
||||||
>
|
>
|
||||||
<form
|
<form
|
||||||
action="/todos/{todo.uid}.json?_method=PATCH"
|
action="/todos?_method=PATCH"
|
||||||
method="post"
|
method="post"
|
||||||
use:enhance={{
|
use:enhance={{
|
||||||
pending: (data) => {
|
pending: ({ data }) => {
|
||||||
todo.done = !!data.get('done');
|
todo.done = !!data.get('done');
|
||||||
},
|
}
|
||||||
result: patch
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
<input type="hidden" name="uid" value={todo.uid} />
|
||||||
<input type="hidden" name="done" value={todo.done ? '' : 'true'} />
|
<input type="hidden" name="done" value={todo.done ? '' : 'true'} />
|
||||||
<button class="toggle" aria-label="Mark todo as {todo.done ? 'not done' : 'done'}" />
|
<button class="toggle" aria-label="Mark todo as {todo.done ? 'not done' : 'done'}" />
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<form
|
<form class="text" action="/todos?_method=PATCH" method="post" use:enhance>
|
||||||
class="text"
|
<input type="hidden" name="uid" value={todo.uid} />
|
||||||
action="/todos/{todo.uid}.json?_method=PATCH"
|
|
||||||
method="post"
|
|
||||||
use:enhance={{
|
|
||||||
result: patch
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<input aria-label="Edit todo" type="text" name="text" value={todo.text} />
|
<input aria-label="Edit todo" type="text" name="text" value={todo.text} />
|
||||||
<button class="save" aria-label="Save todo" />
|
<button class="save" aria-label="Save todo" />
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<form
|
<form
|
||||||
action="/todos/{todo.uid}.json?_method=DELETE"
|
action="/todos?_method=DELETE"
|
||||||
method="post"
|
method="post"
|
||||||
use:enhance={{
|
use:enhance={{
|
||||||
pending: () => (todo.pending_delete = true),
|
pending: () => (todo.pending_delete = true)
|
||||||
result: () => {
|
|
||||||
todos = todos.filter((t) => t.uid !== todo.uid);
|
|
||||||
}
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
<input type="hidden" name="uid" value={todo.uid} />
|
||||||
<button class="delete" aria-label="Delete todo" disabled={todo.pending_delete} />
|
<button class="delete" aria-label="Delete todo" disabled={todo.pending_delete} />
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
@@ -158,7 +116,7 @@
|
|||||||
.done {
|
.done {
|
||||||
transform: none;
|
transform: none;
|
||||||
opacity: 0.4;
|
opacity: 0.4;
|
||||||
filter: drop-shadow(0px 0px 1px rgba(0, 0, 0, 0.1));
|
filter: drop-shadow(0 0 1px rgba(0, 0, 0, 0.1));
|
||||||
}
|
}
|
||||||
|
|
||||||
form.text {
|
form.text {
|
||||||
|
|||||||
@@ -5,8 +5,10 @@ const config = {
|
|||||||
kit: {
|
kit: {
|
||||||
adapter: adapter(),
|
adapter: adapter(),
|
||||||
|
|
||||||
// hydrate the <div id="svelte"> element in src/app.html
|
// Override http methods in the Todo forms
|
||||||
target: '#svelte'
|
methodOverride: {
|
||||||
|
allowed: ['PATCH', 'DELETE']
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
48
package.json
48
package.json
@@ -14,37 +14,24 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"lerna": "3.16.4"
|
"lerna": "3.16.4"
|
||||||
},
|
},
|
||||||
"turbo": {
|
|
||||||
"baseBranch": "origin/main",
|
|
||||||
"pipeline": {
|
|
||||||
"build": {
|
|
||||||
"dependsOn": [
|
|
||||||
"^build"
|
|
||||||
],
|
|
||||||
"outputs": [
|
|
||||||
"dist/**"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@typescript-eslint/eslint-plugin": "4.28.0",
|
"@typescript-eslint/eslint-plugin": "5.21.0",
|
||||||
"@typescript-eslint/parser": "4.28.0",
|
"@typescript-eslint/parser": "5.21.0",
|
||||||
"async-retry": "1.2.3",
|
"async-retry": "1.2.3",
|
||||||
"buffer-replace": "1.0.0",
|
"buffer-replace": "1.0.0",
|
||||||
"cheerio": "1.0.0-rc.3",
|
"eslint": "8.14.0",
|
||||||
"eslint": "7.29.0",
|
"eslint-config-prettier": "8.5.0",
|
||||||
"eslint-config-prettier": "8.3.0",
|
"eslint-plugin-jest": "26.1.5",
|
||||||
"eslint-plugin-jest": "24.3.6",
|
"husky": "7.0.4",
|
||||||
"husky": "6.0.0",
|
"jest": "28.0.2",
|
||||||
"jest": "27.3.1",
|
|
||||||
"json5": "2.1.1",
|
"json5": "2.1.1",
|
||||||
"lint-staged": "9.2.5",
|
"lint-staged": "9.2.5",
|
||||||
"node-fetch": "2.6.1",
|
"node-fetch": "2.6.1",
|
||||||
"npm-package-arg": "6.1.0",
|
"npm-package-arg": "6.1.0",
|
||||||
"prettier": "2.3.1",
|
"prettier": "2.6.2",
|
||||||
"ts-jest": "27.0.4",
|
"ts-eager": "2.0.2",
|
||||||
"turbo": "1.0.18"
|
"ts-jest": "28.0.0-next.1",
|
||||||
|
"turbo": "1.2.5"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"lerna": "lerna",
|
"lerna": "lerna",
|
||||||
@@ -54,9 +41,10 @@
|
|||||||
"publish-from-github": "./utils/publish.sh",
|
"publish-from-github": "./utils/publish.sh",
|
||||||
"changelog": "node utils/changelog.js",
|
"changelog": "node utils/changelog.js",
|
||||||
"build": "turbo run build",
|
"build": "turbo run build",
|
||||||
"vercel-build": "mkdir -p public && echo '<a href=\"https://vercel.com/import\">Import</a>' > public/output.html",
|
"vercel-build": "yarn build && cd api && node -r ts-eager/register ./_lib/script/build.ts",
|
||||||
"pre-commit": "lint-staged",
|
"pre-commit": "lint-staged",
|
||||||
"test-unit": "node utils/run.js test-unit",
|
"test": "jest --rootDir=\"test\" --testPathPattern=\"\\.test.js\"",
|
||||||
|
"test-unit": "yarn test && node utils/run.js test-unit",
|
||||||
"test-integration-cli": "node utils/run.js test-integration-cli",
|
"test-integration-cli": "node utils/run.js test-integration-cli",
|
||||||
"test-integration-once": "node utils/run.js test-integration-once",
|
"test-integration-once": "node utils/run.js test-integration-once",
|
||||||
"test-integration-dev": "node utils/run.js test-integration-dev",
|
"test-integration-dev": "node utils/run.js test-integration-dev",
|
||||||
@@ -102,6 +90,14 @@
|
|||||||
"es6": true
|
"es6": true
|
||||||
},
|
},
|
||||||
"rules": {
|
"rules": {
|
||||||
|
"no-restricted-syntax": [
|
||||||
|
"warn",
|
||||||
|
"WithStatement",
|
||||||
|
{
|
||||||
|
"message": "substr() is deprecated, use slice() or substring() instead",
|
||||||
|
"selector": "MemberExpression > Identifier[name='substr']"
|
||||||
|
}
|
||||||
|
],
|
||||||
"require-atomic-updates": 0,
|
"require-atomic-updates": 0,
|
||||||
"@typescript-eslint/ban-ts-comment": 0,
|
"@typescript-eslint/ban-ts-comment": 0,
|
||||||
"@typescript-eslint/camelcase": 0,
|
"@typescript-eslint/camelcase": 0,
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
/src
|
/src
|
||||||
/test
|
/test
|
||||||
|
/tsconfig.json
|
||||||
|
/.turbo
|
||||||
|
/jest.config.js
|
||||||
tmp
|
tmp
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@vercel/build-utils",
|
"name": "@vercel/build-utils",
|
||||||
"version": "2.14.1-canary.1",
|
"version": "2.16.0",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"main": "./dist/index.js",
|
"main": "./dist/index.js",
|
||||||
"types": "./dist/index.d.js",
|
"types": "./dist/index.d.js",
|
||||||
@@ -23,14 +23,14 @@
|
|||||||
"@types/end-of-stream": "^1.4.0",
|
"@types/end-of-stream": "^1.4.0",
|
||||||
"@types/fs-extra": "9.0.13",
|
"@types/fs-extra": "9.0.13",
|
||||||
"@types/glob": "^7.1.1",
|
"@types/glob": "^7.1.1",
|
||||||
"@types/jest": "27.0.1",
|
"@types/jest": "27.4.1",
|
||||||
"@types/js-yaml": "3.12.1",
|
"@types/js-yaml": "3.12.1",
|
||||||
"@types/ms": "0.7.31",
|
"@types/ms": "0.7.31",
|
||||||
"@types/multistream": "2.1.1",
|
"@types/multistream": "2.1.1",
|
||||||
"@types/node-fetch": "^2.1.6",
|
"@types/node-fetch": "^2.1.6",
|
||||||
"@types/semver": "6.0.0",
|
"@types/semver": "6.0.0",
|
||||||
"@types/yazl": "^2.4.1",
|
"@types/yazl": "2.4.2",
|
||||||
"@vercel/frameworks": "0.6.1-canary.1",
|
"@vercel/frameworks": "0.8.0",
|
||||||
"@vercel/ncc": "0.24.0",
|
"@vercel/ncc": "0.24.0",
|
||||||
"aggregate-error": "3.0.1",
|
"aggregate-error": "3.0.1",
|
||||||
"async-retry": "1.2.3",
|
"async-retry": "1.2.3",
|
||||||
@@ -47,6 +47,6 @@
|
|||||||
"node-fetch": "2.6.1",
|
"node-fetch": "2.6.1",
|
||||||
"semver": "6.1.1",
|
"semver": "6.1.1",
|
||||||
"typescript": "4.3.4",
|
"typescript": "4.3.4",
|
||||||
"yazl": "2.4.3"
|
"yazl": "2.5.1"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -67,8 +67,7 @@ function getPublicBuilder(
|
|||||||
typeof builder.src === 'string' &&
|
typeof builder.src === 'string' &&
|
||||||
isOfficialRuntime('static', builder.use) &&
|
isOfficialRuntime('static', builder.use) &&
|
||||||
/^.*\/\*\*\/\*$/.test(builder.src) &&
|
/^.*\/\*\*\/\*$/.test(builder.src) &&
|
||||||
builder.config &&
|
builder.config?.zeroConfig === true
|
||||||
builder.config.zeroConfig === true
|
|
||||||
) {
|
) {
|
||||||
return builder as Builder & { src: string };
|
return builder as Builder & { src: string };
|
||||||
}
|
}
|
||||||
@@ -539,7 +538,7 @@ function getMissingBuildScriptError() {
|
|||||||
code: 'missing_build_script',
|
code: 'missing_build_script',
|
||||||
message:
|
message:
|
||||||
'Your `package.json` file is missing a `build` property inside the `scripts` property.' +
|
'Your `package.json` file is missing a `build` property inside the `scripts` property.' +
|
||||||
'\nLearn More: https://vercel.com/docs/v2/platform/frequently-asked-questions#missing-build-script',
|
'\nLearn More: https://vercel.link/missing-build-script',
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
44
packages/build-utils/src/edge-function.ts
Normal file
44
packages/build-utils/src/edge-function.ts
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import type { Files } from './types';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An Edge Functions output
|
||||||
|
*/
|
||||||
|
export class EdgeFunction {
|
||||||
|
type: 'EdgeFunction';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A display name for the edge function.
|
||||||
|
*/
|
||||||
|
name: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The deployment target.
|
||||||
|
* Only v8-worker is currently supported.
|
||||||
|
*/
|
||||||
|
deploymentTarget: 'v8-worker';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The entrypoint for the edge function.
|
||||||
|
*/
|
||||||
|
entrypoint: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The list of files to be included in the edge function bundle.
|
||||||
|
*/
|
||||||
|
files: Files;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extra environment variables in use for the user code, to be
|
||||||
|
* assigned to the edge function.
|
||||||
|
*/
|
||||||
|
envVarsInUse?: string[];
|
||||||
|
|
||||||
|
constructor(params: Omit<EdgeFunction, 'type'>) {
|
||||||
|
this.type = 'EdgeFunction';
|
||||||
|
this.name = params.name;
|
||||||
|
this.deploymentTarget = params.deploymentTarget;
|
||||||
|
this.entrypoint = params.entrypoint;
|
||||||
|
this.files = params.files;
|
||||||
|
this.envVarsInUse = params.envVarsInUse;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import assert from 'assert';
|
import assert from 'assert';
|
||||||
import intoStream from 'into-stream';
|
import intoStream from 'into-stream';
|
||||||
import { File } from './types';
|
import { FileBase } from './types';
|
||||||
|
|
||||||
interface FileBlobOptions {
|
interface FileBlobOptions {
|
||||||
mode?: number;
|
mode?: number;
|
||||||
@@ -14,7 +14,7 @@ interface FromStreamOptions {
|
|||||||
stream: NodeJS.ReadableStream;
|
stream: NodeJS.ReadableStream;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class FileBlob implements File {
|
export default class FileBlob implements FileBase {
|
||||||
public type: 'FileBlob';
|
public type: 'FileBlob';
|
||||||
public mode: number;
|
public mode: number;
|
||||||
public data: string | Buffer;
|
public data: string | Buffer;
|
||||||
@@ -48,6 +48,10 @@ export default class FileBlob implements File {
|
|||||||
return new FileBlob({ mode, contentType, data });
|
return new FileBlob({ mode, contentType, data });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async toStreamAsync(): Promise<NodeJS.ReadableStream> {
|
||||||
|
return this.toStream();
|
||||||
|
}
|
||||||
|
|
||||||
toStream(): NodeJS.ReadableStream {
|
toStream(): NodeJS.ReadableStream {
|
||||||
return intoStream(this.data);
|
return intoStream(this.data);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import fs from 'fs-extra';
|
|||||||
import multiStream from 'multistream';
|
import multiStream from 'multistream';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import Sema from 'async-sema';
|
import Sema from 'async-sema';
|
||||||
import { File } from './types';
|
import { FileBase } from './types';
|
||||||
|
|
||||||
const semaToPreventEMFILE = new Sema(20);
|
const semaToPreventEMFILE = new Sema(20);
|
||||||
|
|
||||||
@@ -20,7 +20,7 @@ interface FromStreamOptions {
|
|||||||
fsPath: string;
|
fsPath: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
class FileFsRef implements File {
|
class FileFsRef implements FileBase {
|
||||||
public type: 'FileFsRef';
|
public type: 'FileFsRef';
|
||||||
public mode: number;
|
public mode: number;
|
||||||
public fsPath: string;
|
public fsPath: string;
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import fetch from 'node-fetch';
|
|||||||
import multiStream from 'multistream';
|
import multiStream from 'multistream';
|
||||||
import retry from 'async-retry';
|
import retry from 'async-retry';
|
||||||
import Sema from 'async-sema';
|
import Sema from 'async-sema';
|
||||||
import { File } from './types';
|
import { FileBase } from './types';
|
||||||
|
|
||||||
interface FileRefOptions {
|
interface FileRefOptions {
|
||||||
mode?: number;
|
mode?: number;
|
||||||
@@ -23,7 +23,7 @@ class BailableError extends Error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class FileRef implements File {
|
export default class FileRef implements FileBase {
|
||||||
public type: 'FileRef';
|
public type: 'FileRef';
|
||||||
public mode: number;
|
public mode: number;
|
||||||
public digest: string;
|
public digest: string;
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import debug from '../debug';
|
|||||||
import FileFsRef from '../file-fs-ref';
|
import FileFsRef from '../file-fs-ref';
|
||||||
import { File, Files, Meta } from '../types';
|
import { File, Files, Meta } from '../types';
|
||||||
import { remove, mkdirp, readlink, symlink } from 'fs-extra';
|
import { remove, mkdirp, readlink, symlink } from 'fs-extra';
|
||||||
|
import streamToBuffer from './stream-to-buffer';
|
||||||
|
|
||||||
export interface DownloadedFiles {
|
export interface DownloadedFiles {
|
||||||
[filePath: string]: FileFsRef;
|
[filePath: string]: FileFsRef;
|
||||||
@@ -15,20 +16,45 @@ export function isSymbolicLink(mode: number): boolean {
|
|||||||
return (mode & S_IFMT) === S_IFLNK;
|
return (mode & S_IFMT) === S_IFLNK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function prepareSymlinkTarget(
|
||||||
|
file: File,
|
||||||
|
fsPath: string
|
||||||
|
): Promise<string> {
|
||||||
|
const mkdirPromise = mkdirp(path.dirname(fsPath));
|
||||||
|
if (file.type === 'FileFsRef') {
|
||||||
|
const [target] = await Promise.all([readlink(file.fsPath), mkdirPromise]);
|
||||||
|
return target;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (file.type === 'FileRef' || file.type === 'FileBlob') {
|
||||||
|
const targetPathBufferPromise = await streamToBuffer(
|
||||||
|
await file.toStreamAsync()
|
||||||
|
);
|
||||||
|
const [targetPathBuffer] = await Promise.all([
|
||||||
|
targetPathBufferPromise,
|
||||||
|
mkdirPromise,
|
||||||
|
]);
|
||||||
|
return targetPathBuffer.toString('utf8');
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error(
|
||||||
|
`file.type "${(file as any).type}" not supported for symlink`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
async function downloadFile(file: File, fsPath: string): Promise<FileFsRef> {
|
async function downloadFile(file: File, fsPath: string): Promise<FileFsRef> {
|
||||||
const { mode } = file;
|
const { mode } = file;
|
||||||
if (mode && isSymbolicLink(mode) && file.type === 'FileFsRef') {
|
|
||||||
const [target] = await Promise.all([
|
if (isSymbolicLink(mode)) {
|
||||||
readlink((file as FileFsRef).fsPath),
|
const target = await prepareSymlinkTarget(file, fsPath);
|
||||||
mkdirp(path.dirname(fsPath)),
|
|
||||||
]);
|
|
||||||
await symlink(target, fsPath);
|
await symlink(target, fsPath);
|
||||||
return FileFsRef.fromFsPath({ mode, fsPath });
|
return FileFsRef.fromFsPath({ mode, fsPath });
|
||||||
} else {
|
}
|
||||||
|
|
||||||
const stream = file.toStream();
|
const stream = file.toStream();
|
||||||
return FileFsRef.fromStream({ mode, stream, fsPath });
|
return FileFsRef.fromStream({ mode, stream, fsPath });
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
async function removeFile(basePath: string, fileMatched: string) {
|
async function removeFile(basePath: string, fileMatched: string) {
|
||||||
const file = path.join(basePath, fileMatched);
|
const file = path.join(basePath, fileMatched);
|
||||||
|
|||||||
@@ -46,8 +46,8 @@ export default async function glob(
|
|||||||
const files = await vanillaGlob(pattern, options);
|
const files = await vanillaGlob(pattern, options);
|
||||||
|
|
||||||
for (const relativePath of files) {
|
for (const relativePath of files) {
|
||||||
const fsPath = normalizePath(path.join(options.cwd!, relativePath));
|
const fsPath = normalizePath(path.join(options.cwd, relativePath));
|
||||||
let stat: Stats = options.statCache![fsPath] as Stats;
|
let stat: Stats = options.statCache[fsPath] as Stats;
|
||||||
assert(
|
assert(
|
||||||
stat,
|
stat,
|
||||||
`statCache does not contain value for ${relativePath} (resolved to ${fsPath})`
|
`statCache does not contain value for ${relativePath} (resolved to ${fsPath})`
|
||||||
|
|||||||
@@ -1,21 +1,31 @@
|
|||||||
import assert from 'assert';
|
import assert from 'assert';
|
||||||
import fs from 'fs-extra';
|
import fs from 'fs-extra';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import debug from '../debug';
|
import Sema from 'async-sema';
|
||||||
import spawn from 'cross-spawn';
|
import spawn from 'cross-spawn';
|
||||||
import { SpawnOptions } from 'child_process';
|
import { SpawnOptions } from 'child_process';
|
||||||
import { deprecate } from 'util';
|
import { deprecate } from 'util';
|
||||||
|
import debug from '../debug';
|
||||||
import { NowBuildError } from '../errors';
|
import { NowBuildError } from '../errors';
|
||||||
import { Meta, PackageJson, NodeVersion, Config } from '../types';
|
import { Meta, PackageJson, NodeVersion, Config } from '../types';
|
||||||
import { getSupportedNodeVersion, getLatestNodeVersion } from './node-version';
|
import { getSupportedNodeVersion, getLatestNodeVersion } from './node-version';
|
||||||
|
import { readConfigFile } from './read-config-file';
|
||||||
|
|
||||||
export type CliType = 'yarn' | 'npm';
|
// Only allow one `runNpmInstall()` invocation to run concurrently
|
||||||
|
const runNpmInstallSema = new Sema(1);
|
||||||
|
|
||||||
|
export type CliType = 'yarn' | 'npm' | 'pnpm';
|
||||||
|
|
||||||
export interface ScanParentDirsResult {
|
export interface ScanParentDirsResult {
|
||||||
/**
|
/**
|
||||||
* "yarn" or "npm", depending on the presence of lockfiles.
|
* "yarn", "npm", or "pnpm" depending on the presence of lockfiles.
|
||||||
*/
|
*/
|
||||||
cliType: CliType;
|
cliType: CliType;
|
||||||
|
/**
|
||||||
|
* The file path of found `package.json` file, or `undefined` if none was
|
||||||
|
* found.
|
||||||
|
*/
|
||||||
|
packageJsonPath?: string;
|
||||||
/**
|
/**
|
||||||
* The contents of found `package.json` file, when the `readPackageJson`
|
* The contents of found `package.json` file, when the `readPackageJson`
|
||||||
* option is enabled.
|
* option is enabled.
|
||||||
@@ -212,6 +222,12 @@ export async function getNodeVersion(
|
|||||||
const latest = getLatestNodeVersion();
|
const latest = getLatestNodeVersion();
|
||||||
return { ...latest, runtime: 'nodejs' };
|
return { ...latest, runtime: 'nodejs' };
|
||||||
}
|
}
|
||||||
|
if (process.env.ENABLE_EXPERIMENTAL_NODE16 === '1') {
|
||||||
|
console.warn(
|
||||||
|
'Warning: Using experimental Node.js 16.x due to ENABLE_EXPERIMENTAL_NODE16=1'
|
||||||
|
);
|
||||||
|
return { major: 16, range: '16.x', runtime: 'nodejs16.x' };
|
||||||
|
}
|
||||||
const { packageJson } = await scanParentDirs(destPath, true);
|
const { packageJson } = await scanParentDirs(destPath, true);
|
||||||
let { nodeVersion } = config;
|
let { nodeVersion } = config;
|
||||||
let isAuto = true;
|
let isAuto = true;
|
||||||
@@ -236,12 +252,13 @@ export async function scanParentDirs(
|
|||||||
|
|
||||||
let cliType: CliType = 'yarn';
|
let cliType: CliType = 'yarn';
|
||||||
let packageJson: PackageJson | undefined;
|
let packageJson: PackageJson | undefined;
|
||||||
|
let packageJsonPath: string | undefined;
|
||||||
let currentDestPath = destPath;
|
let currentDestPath = destPath;
|
||||||
let lockfileVersion: number | undefined;
|
let lockfileVersion: number | undefined;
|
||||||
|
|
||||||
// eslint-disable-next-line no-constant-condition
|
// eslint-disable-next-line no-constant-condition
|
||||||
while (true) {
|
while (true) {
|
||||||
const packageJsonPath = path.join(currentDestPath, 'package.json');
|
packageJsonPath = path.join(currentDestPath, 'package.json');
|
||||||
// eslint-disable-next-line no-await-in-loop
|
// eslint-disable-next-line no-await-in-loop
|
||||||
if (await fs.pathExists(packageJsonPath)) {
|
if (await fs.pathExists(packageJsonPath)) {
|
||||||
// Only read the contents of the *first* `package.json` file found,
|
// Only read the contents of the *first* `package.json` file found,
|
||||||
@@ -252,7 +269,7 @@ export async function scanParentDirs(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-await-in-loop
|
// eslint-disable-next-line no-await-in-loop
|
||||||
const [packageLockJson, hasYarnLock] = await Promise.all([
|
const [packageLockJson, hasYarnLock, pnpmLockYaml] = await Promise.all([
|
||||||
fs
|
fs
|
||||||
.readJson(path.join(currentDestPath, 'package-lock.json'))
|
.readJson(path.join(currentDestPath, 'package-lock.json'))
|
||||||
.catch(error => {
|
.catch(error => {
|
||||||
@@ -263,17 +280,26 @@ export async function scanParentDirs(
|
|||||||
throw error;
|
throw error;
|
||||||
}),
|
}),
|
||||||
fs.pathExists(path.join(currentDestPath, 'yarn.lock')),
|
fs.pathExists(path.join(currentDestPath, 'yarn.lock')),
|
||||||
|
readConfigFile<{ lockfileVersion: number }>(
|
||||||
|
path.join(currentDestPath, 'pnpm-lock.yaml')
|
||||||
|
),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (packageLockJson && !hasYarnLock) {
|
if (packageLockJson && !hasYarnLock && !pnpmLockYaml) {
|
||||||
cliType = 'npm';
|
cliType = 'npm';
|
||||||
lockfileVersion = packageLockJson.lockfileVersion;
|
lockfileVersion = packageLockJson.lockfileVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!packageLockJson && !hasYarnLock && pnpmLockYaml) {
|
||||||
|
cliType = 'pnpm';
|
||||||
|
// just ensure that it is read as a number and not a string
|
||||||
|
lockfileVersion = Number(pnpmLockYaml.lockfileVersion);
|
||||||
|
}
|
||||||
|
|
||||||
// Only stop iterating if a lockfile was found, because it's possible
|
// Only stop iterating if a lockfile was found, because it's possible
|
||||||
// that the lockfile is in a higher path than where the `package.json`
|
// that the lockfile is in a higher path than where the `package.json`
|
||||||
// file was found.
|
// file was found.
|
||||||
if (packageLockJson || hasYarnLock) {
|
if (packageLockJson || hasYarnLock || pnpmLockYaml) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -283,7 +309,7 @@ export async function scanParentDirs(
|
|||||||
currentDestPath = newDestPath;
|
currentDestPath = newDestPath;
|
||||||
}
|
}
|
||||||
|
|
||||||
return { cliType, packageJson, lockfileVersion };
|
return { cliType, packageJson, lockfileVersion, packageJsonPath };
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function walkParentDirs({
|
export async function walkParentDirs({
|
||||||
@@ -309,22 +335,49 @@ export async function walkParentDirs({
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function isSet<T>(v: any): v is Set<T> {
|
||||||
|
return v?.constructor?.name === 'Set';
|
||||||
|
}
|
||||||
|
|
||||||
export async function runNpmInstall(
|
export async function runNpmInstall(
|
||||||
destPath: string,
|
destPath: string,
|
||||||
args: string[] = [],
|
args: string[] = [],
|
||||||
spawnOpts?: SpawnOptions,
|
spawnOpts?: SpawnOptions,
|
||||||
meta?: Meta,
|
meta?: Meta,
|
||||||
nodeVersion?: NodeVersion
|
nodeVersion?: NodeVersion
|
||||||
) {
|
): Promise<boolean> {
|
||||||
if (meta?.isDev) {
|
if (meta?.isDev) {
|
||||||
debug('Skipping dependency installation because dev mode is enabled');
|
debug('Skipping dependency installation because dev mode is enabled');
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
assert(path.isAbsolute(destPath));
|
assert(path.isAbsolute(destPath));
|
||||||
|
|
||||||
|
try {
|
||||||
|
await runNpmInstallSema.acquire();
|
||||||
|
const { cliType, packageJsonPath, lockfileVersion } = await scanParentDirs(
|
||||||
|
destPath
|
||||||
|
);
|
||||||
|
|
||||||
|
// Only allow `runNpmInstall()` to run once per `package.json`
|
||||||
|
// when doing a default install (no additional args)
|
||||||
|
if (meta && packageJsonPath && args.length === 0) {
|
||||||
|
if (!isSet<string>(meta.runNpmInstallSet)) {
|
||||||
|
meta.runNpmInstallSet = new Set<string>();
|
||||||
|
}
|
||||||
|
if (isSet<string>(meta.runNpmInstallSet)) {
|
||||||
|
if (meta.runNpmInstallSet.has(packageJsonPath)) {
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
meta.runNpmInstallSet.add(packageJsonPath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const installTime = Date.now();
|
||||||
|
console.log('Installing dependencies...');
|
||||||
debug(`Installing to ${destPath}`);
|
debug(`Installing to ${destPath}`);
|
||||||
|
|
||||||
const { cliType, lockfileVersion } = await scanParentDirs(destPath);
|
|
||||||
const opts: SpawnOptionsExtended = { cwd: destPath, ...spawnOpts };
|
const opts: SpawnOptionsExtended = { cwd: destPath, ...spawnOpts };
|
||||||
const env = opts.env ? { ...opts.env } : { ...process.env };
|
const env = opts.env ? { ...opts.env } : { ...process.env };
|
||||||
delete env.NODE_ENV;
|
delete env.NODE_ENV;
|
||||||
@@ -341,6 +394,13 @@ export async function runNpmInstall(
|
|||||||
commandArgs = args
|
commandArgs = args
|
||||||
.filter(a => a !== '--prefer-offline')
|
.filter(a => a !== '--prefer-offline')
|
||||||
.concat(['install', '--no-audit', '--unsafe-perm']);
|
.concat(['install', '--no-audit', '--unsafe-perm']);
|
||||||
|
} else if (cliType === 'pnpm') {
|
||||||
|
// PNPM's install command is similar to NPM's but without the audit nonsense
|
||||||
|
// @see options https://pnpm.io/cli/install
|
||||||
|
opts.prettyCommand = 'pnpm install';
|
||||||
|
commandArgs = args
|
||||||
|
.filter(a => a !== '--prefer-offline')
|
||||||
|
.concat(['install', '--unsafe-perm']);
|
||||||
} else {
|
} else {
|
||||||
opts.prettyCommand = 'yarn install';
|
opts.prettyCommand = 'yarn install';
|
||||||
commandArgs = ['install', ...args];
|
commandArgs = ['install', ...args];
|
||||||
@@ -350,7 +410,12 @@ export async function runNpmInstall(
|
|||||||
commandArgs.push('--production');
|
commandArgs.push('--production');
|
||||||
}
|
}
|
||||||
|
|
||||||
return spawnAsync(cliType, commandArgs, opts);
|
await spawnAsync(cliType, commandArgs, opts);
|
||||||
|
debug(`Install complete [${Date.now() - installTime}ms]`);
|
||||||
|
return true;
|
||||||
|
} finally {
|
||||||
|
runNpmInstallSema.release();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getEnvForPackageManager({
|
export function getEnvForPackageManager({
|
||||||
@@ -448,6 +513,8 @@ export async function runPackageJsonScript(
|
|||||||
|
|
||||||
if (cliType === 'npm') {
|
if (cliType === 'npm') {
|
||||||
opts.prettyCommand = `npm run ${scriptName}`;
|
opts.prettyCommand = `npm run ${scriptName}`;
|
||||||
|
} else if (cliType === 'pnpm') {
|
||||||
|
opts.prettyCommand = `pnpm run ${scriptName}`;
|
||||||
} else {
|
} else {
|
||||||
opts.prettyCommand = `yarn run ${scriptName}`;
|
opts.prettyCommand = `yarn run ${scriptName}`;
|
||||||
}
|
}
|
||||||
@@ -483,7 +550,7 @@ export async function runPipInstall(
|
|||||||
meta?: Meta
|
meta?: Meta
|
||||||
) {
|
) {
|
||||||
if (meta && meta.isDev) {
|
if (meta && meta.isDev) {
|
||||||
debug('Skipping dependency installation because dev mode is enabled');
|
debug('Skipping dependency installation because dev mode is enabled');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -81,6 +81,7 @@ export {
|
|||||||
getIgnoreFilter,
|
getIgnoreFilter,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export { EdgeFunction } from './edge-function';
|
||||||
export {
|
export {
|
||||||
detectBuilders,
|
detectBuilders,
|
||||||
detectOutputDirectory,
|
detectOutputDirectory,
|
||||||
|
|||||||
@@ -3,17 +3,17 @@ import Sema from 'async-sema';
|
|||||||
import { ZipFile } from 'yazl';
|
import { ZipFile } from 'yazl';
|
||||||
import minimatch from 'minimatch';
|
import minimatch from 'minimatch';
|
||||||
import { readlink } from 'fs-extra';
|
import { readlink } from 'fs-extra';
|
||||||
import { Files, Config } from './types';
|
|
||||||
import FileFsRef from './file-fs-ref';
|
|
||||||
import { isSymbolicLink } from './fs/download';
|
import { isSymbolicLink } from './fs/download';
|
||||||
import streamToBuffer from './fs/stream-to-buffer';
|
import streamToBuffer from './fs/stream-to-buffer';
|
||||||
|
import type { Files, Config } from './types';
|
||||||
|
|
||||||
interface Environment {
|
interface Environment {
|
||||||
[key: string]: string;
|
[key: string]: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface LambdaOptions {
|
export type LambdaOptions = LambdaOptionsWithFiles | LambdaOptionsWithZipBuffer;
|
||||||
files: Files;
|
|
||||||
|
export interface LambdaOptionsBase {
|
||||||
handler: string;
|
handler: string;
|
||||||
runtime: string;
|
runtime: string;
|
||||||
memory?: number;
|
memory?: number;
|
||||||
@@ -21,10 +21,21 @@ export interface LambdaOptions {
|
|||||||
environment?: Environment;
|
environment?: Environment;
|
||||||
allowQuery?: string[];
|
allowQuery?: string[];
|
||||||
regions?: string[];
|
regions?: string[];
|
||||||
|
supportsMultiPayloads?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LambdaOptionsWithFiles extends LambdaOptionsBase {
|
||||||
|
files: Files;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use `LambdaOptionsWithFiles` instead.
|
||||||
|
*/
|
||||||
|
export interface LambdaOptionsWithZipBuffer extends LambdaOptionsBase {
|
||||||
/**
|
/**
|
||||||
* @deprecated Use `files` property instead.
|
* @deprecated Use `files` property instead.
|
||||||
*/
|
*/
|
||||||
zipBuffer?: Buffer;
|
zipBuffer: Buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface GetLambdaOptionsFromFunctionOptions {
|
interface GetLambdaOptionsFromFunctionOptions {
|
||||||
@@ -34,7 +45,7 @@ interface GetLambdaOptionsFromFunctionOptions {
|
|||||||
|
|
||||||
export class Lambda {
|
export class Lambda {
|
||||||
type: 'Lambda';
|
type: 'Lambda';
|
||||||
files: Files;
|
files?: Files;
|
||||||
handler: string;
|
handler: string;
|
||||||
runtime: string;
|
runtime: string;
|
||||||
memory?: number;
|
memory?: number;
|
||||||
@@ -46,9 +57,10 @@ export class Lambda {
|
|||||||
* @deprecated Use `await lambda.createZip()` instead.
|
* @deprecated Use `await lambda.createZip()` instead.
|
||||||
*/
|
*/
|
||||||
zipBuffer?: Buffer;
|
zipBuffer?: Buffer;
|
||||||
|
supportsMultiPayloads?: boolean;
|
||||||
|
|
||||||
constructor({
|
constructor(opts: LambdaOptions) {
|
||||||
files,
|
const {
|
||||||
handler,
|
handler,
|
||||||
runtime,
|
runtime,
|
||||||
maxDuration,
|
maxDuration,
|
||||||
@@ -56,10 +68,13 @@ export class Lambda {
|
|||||||
environment = {},
|
environment = {},
|
||||||
allowQuery,
|
allowQuery,
|
||||||
regions,
|
regions,
|
||||||
zipBuffer,
|
supportsMultiPayloads,
|
||||||
}: LambdaOptions) {
|
} = opts;
|
||||||
if (!zipBuffer) {
|
if ('files' in opts) {
|
||||||
assert(typeof files === 'object', '"files" must be an object');
|
assert(typeof opts.files === 'object', '"files" must be an object');
|
||||||
|
}
|
||||||
|
if ('zipBuffer' in opts) {
|
||||||
|
assert(Buffer.isBuffer(opts.zipBuffer), '"zipBuffer" must be a Buffer');
|
||||||
}
|
}
|
||||||
assert(typeof handler === 'string', '"handler" is not a string');
|
assert(typeof handler === 'string', '"handler" is not a string');
|
||||||
assert(typeof runtime === 'string', '"runtime" is not a string');
|
assert(typeof runtime === 'string', '"runtime" is not a string');
|
||||||
@@ -81,6 +96,13 @@ export class Lambda {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (supportsMultiPayloads !== undefined) {
|
||||||
|
assert(
|
||||||
|
typeof supportsMultiPayloads === 'boolean',
|
||||||
|
'"supportsMultiPayloads" is not a boolean'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (regions !== undefined) {
|
if (regions !== undefined) {
|
||||||
assert(Array.isArray(regions), '"regions" is not an Array');
|
assert(Array.isArray(regions), '"regions" is not an Array');
|
||||||
assert(
|
assert(
|
||||||
@@ -89,7 +111,7 @@ export class Lambda {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
this.type = 'Lambda';
|
this.type = 'Lambda';
|
||||||
this.files = files;
|
this.files = 'files' in opts ? opts.files : undefined;
|
||||||
this.handler = handler;
|
this.handler = handler;
|
||||||
this.runtime = runtime;
|
this.runtime = runtime;
|
||||||
this.memory = memory;
|
this.memory = memory;
|
||||||
@@ -97,12 +119,16 @@ export class Lambda {
|
|||||||
this.environment = environment;
|
this.environment = environment;
|
||||||
this.allowQuery = allowQuery;
|
this.allowQuery = allowQuery;
|
||||||
this.regions = regions;
|
this.regions = regions;
|
||||||
this.zipBuffer = zipBuffer;
|
this.zipBuffer = 'zipBuffer' in opts ? opts.zipBuffer : undefined;
|
||||||
|
this.supportsMultiPayloads = supportsMultiPayloads;
|
||||||
}
|
}
|
||||||
|
|
||||||
async createZip(): Promise<Buffer> {
|
async createZip(): Promise<Buffer> {
|
||||||
let { zipBuffer } = this;
|
let { zipBuffer } = this;
|
||||||
if (!zipBuffer) {
|
if (!zipBuffer) {
|
||||||
|
if (!this.files) {
|
||||||
|
throw new Error('`files` is not defined');
|
||||||
|
}
|
||||||
await sema.acquire();
|
await sema.acquire();
|
||||||
try {
|
try {
|
||||||
zipBuffer = await createZip(this.files);
|
zipBuffer = await createZip(this.files);
|
||||||
@@ -136,7 +162,7 @@ export async function createZip(files: Files): Promise<Buffer> {
|
|||||||
for (const name of names) {
|
for (const name of names) {
|
||||||
const file = files[name];
|
const file = files[name];
|
||||||
if (file.mode && isSymbolicLink(file.mode) && file.type === 'FileFsRef') {
|
if (file.mode && isSymbolicLink(file.mode) && file.type === 'FileFsRef') {
|
||||||
const symlinkTarget = await readlink((file as FileFsRef).fsPath);
|
const symlinkTarget = await readlink(file.fsPath);
|
||||||
symlinkTargets.set(name, symlinkTarget);
|
symlinkTargets.set(name, symlinkTarget);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -150,7 +176,7 @@ export async function createZip(files: Files): Promise<Buffer> {
|
|||||||
if (typeof symlinkTarget === 'string') {
|
if (typeof symlinkTarget === 'string') {
|
||||||
zipFile.addBuffer(Buffer.from(symlinkTarget, 'utf8'), name, opts);
|
zipFile.addBuffer(Buffer.from(symlinkTarget, 'utf8'), name, opts);
|
||||||
} else {
|
} else {
|
||||||
const stream = file.toStream() as import('stream').Readable;
|
const stream = file.toStream();
|
||||||
stream.on('error', reject);
|
stream.on('error', reject);
|
||||||
zipFile.addReadStream(stream, name, opts);
|
zipFile.addReadStream(stream, name, opts);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { Lambda, LambdaOptions } from './lambda';
|
import { Lambda, LambdaOptionsWithFiles } from './lambda';
|
||||||
|
|
||||||
interface NodejsLambdaOptions extends LambdaOptions {
|
interface NodejsLambdaOptions extends LambdaOptionsWithFiles {
|
||||||
shouldAddHelpers: boolean;
|
shouldAddHelpers: boolean;
|
||||||
shouldAddSourcemapSupport: boolean;
|
shouldAddSourcemapSupport: boolean;
|
||||||
awsLambdaHandler?: string;
|
awsLambdaHandler?: string;
|
||||||
|
|||||||
@@ -1,12 +1,10 @@
|
|||||||
import FileBlob from './file-blob';
|
import { File } from './types';
|
||||||
import FileFsRef from './file-fs-ref';
|
|
||||||
import FileRef from './file-ref';
|
|
||||||
import { Lambda } from './lambda';
|
import { Lambda } from './lambda';
|
||||||
|
|
||||||
interface PrerenderOptions {
|
interface PrerenderOptions {
|
||||||
expiration: number | false;
|
expiration: number | false;
|
||||||
lambda: Lambda;
|
lambda: Lambda;
|
||||||
fallback: FileBlob | FileFsRef | FileRef | null;
|
fallback: File | null;
|
||||||
group?: number;
|
group?: number;
|
||||||
bypassToken?: string | null /* optional to be non-breaking change */;
|
bypassToken?: string | null /* optional to be non-breaking change */;
|
||||||
allowQuery?: string[];
|
allowQuery?: string[];
|
||||||
@@ -16,7 +14,7 @@ export class Prerender {
|
|||||||
public type: 'Prerender';
|
public type: 'Prerender';
|
||||||
public expiration: number | false;
|
public expiration: number | false;
|
||||||
public lambda: Lambda;
|
public lambda: Lambda;
|
||||||
public fallback: FileBlob | FileFsRef | FileRef | null;
|
public fallback: File | null;
|
||||||
public group?: number;
|
public group?: number;
|
||||||
public bypassToken: string | null;
|
public bypassToken: string | null;
|
||||||
public allowQuery?: string[];
|
public allowQuery?: string[];
|
||||||
|
|||||||
@@ -1,21 +1,21 @@
|
|||||||
import FileRef from './file-ref';
|
import type FileRef from './file-ref';
|
||||||
import FileFsRef from './file-fs-ref';
|
import type FileFsRef from './file-fs-ref';
|
||||||
import { Lambda } from './lambda';
|
import type FileBlob from './file-blob';
|
||||||
|
import type { Lambda } from './lambda';
|
||||||
|
import type { Prerender } from './prerender';
|
||||||
|
import type { EdgeFunction } from './edge-function';
|
||||||
|
|
||||||
export interface Env {
|
export interface Env {
|
||||||
[name: string]: string | undefined;
|
[name: string]: string | undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface File {
|
export type File = FileRef | FileFsRef | FileBlob;
|
||||||
|
export interface FileBase {
|
||||||
type: string;
|
type: string;
|
||||||
mode: number;
|
mode: number;
|
||||||
contentType?: string;
|
contentType?: string;
|
||||||
toStream: () => NodeJS.ReadableStream;
|
toStream: () => NodeJS.ReadableStream;
|
||||||
toStreamAsync?: () => Promise<NodeJS.ReadableStream>;
|
toStreamAsync?: () => Promise<NodeJS.ReadableStream>;
|
||||||
/**
|
|
||||||
* The absolute path to the file in the filesystem
|
|
||||||
*/
|
|
||||||
fsPath?: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Files {
|
export interface Files {
|
||||||
@@ -23,16 +23,6 @@ export interface Files {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Config {
|
export interface Config {
|
||||||
[key: string]:
|
|
||||||
| string
|
|
||||||
| string[]
|
|
||||||
| boolean
|
|
||||||
| number
|
|
||||||
| { [key: string]: string }
|
|
||||||
| BuilderFunctions
|
|
||||||
| ProjectSettings
|
|
||||||
| undefined
|
|
||||||
| null;
|
|
||||||
maxLambdaSize?: string;
|
maxLambdaSize?: string;
|
||||||
includeFiles?: string | string[];
|
includeFiles?: string | string[];
|
||||||
excludeFiles?: string | string[];
|
excludeFiles?: string | string[];
|
||||||
@@ -51,6 +41,7 @@ export interface Config {
|
|||||||
devCommand?: string;
|
devCommand?: string;
|
||||||
framework?: string | null;
|
framework?: string | null;
|
||||||
nodeVersion?: string;
|
nodeVersion?: string;
|
||||||
|
[key: string]: unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Meta {
|
export interface Meta {
|
||||||
@@ -63,35 +54,7 @@ export interface Meta {
|
|||||||
env?: Env;
|
env?: Env;
|
||||||
buildEnv?: Env;
|
buildEnv?: Env;
|
||||||
avoidTopLevelInstall?: boolean;
|
avoidTopLevelInstall?: boolean;
|
||||||
}
|
[key: string]: unknown;
|
||||||
|
|
||||||
export interface AnalyzeOptions {
|
|
||||||
/**
|
|
||||||
* All source files of the project
|
|
||||||
*/
|
|
||||||
files: {
|
|
||||||
[filePath: string]: FileRef;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Name of entrypoint file for this particular build job. Value
|
|
||||||
* `files[entrypoint]` is guaranteed to exist and be a valid File reference.
|
|
||||||
* `entrypoint` is always a discrete file and never a glob, since globs are
|
|
||||||
* expanded into separate builds at deployment time.
|
|
||||||
*/
|
|
||||||
entrypoint: string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A writable temporary directory where you are encouraged to perform your
|
|
||||||
* build process. This directory will be populated with the restored cache.
|
|
||||||
*/
|
|
||||||
workPath: string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An arbitrary object passed by the user in the build definition defined
|
|
||||||
* in `vercel.json`.
|
|
||||||
*/
|
|
||||||
config: Config;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface BuildOptions {
|
export interface BuildOptions {
|
||||||
@@ -156,10 +119,11 @@ export interface PrepareCacheOptions {
|
|||||||
workPath: string;
|
workPath: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A writable temporary directory where you can build a cache to use for
|
* The "Root Directory" is assigned to the `workPath` so the `repoRootPath`
|
||||||
* the next run.
|
* is the Git Repository Root. This is only relevant for Monorepos.
|
||||||
|
* See https://vercel.com/blog/monorepos
|
||||||
*/
|
*/
|
||||||
cachePath: string;
|
repoRootPath?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An arbitrary object passed by the user in the build definition defined
|
* An arbitrary object passed by the user in the build definition defined
|
||||||
@@ -368,6 +332,7 @@ export interface ProjectSettings {
|
|||||||
sourceFilesOutsideRootDirectory?: boolean;
|
sourceFilesOutsideRootDirectory?: boolean;
|
||||||
directoryListing?: boolean;
|
directoryListing?: boolean;
|
||||||
gitForkProtection?: boolean;
|
gitForkProtection?: boolean;
|
||||||
|
commandForIgnoringBuildStep?: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface BuilderV2 {
|
export interface BuilderV2 {
|
||||||
@@ -392,12 +357,34 @@ export interface Images {
|
|||||||
formats?: ImageFormat[];
|
formats?: ImageFormat[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface BuildResultV2 {
|
/**
|
||||||
|
* If a Builder ends up creating filesystem outputs conforming to
|
||||||
|
* the Build Output API, then the Builder should return this type.
|
||||||
|
*/
|
||||||
|
export interface BuildResultBuildOutput {
|
||||||
|
/**
|
||||||
|
* Version number of the Build Output API that was created.
|
||||||
|
* Currently only `3` is a valid value.
|
||||||
|
* @example 3
|
||||||
|
*/
|
||||||
|
buildOutputVersion: 3;
|
||||||
|
/**
|
||||||
|
* Filesystem path to the Build Output directory.
|
||||||
|
* @example "/path/to/.vercel/output"
|
||||||
|
*/
|
||||||
|
buildOutputPath: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When a Builder implements `version: 2`, the `build()` function is expected
|
||||||
|
* to return this type.
|
||||||
|
*/
|
||||||
|
export interface BuildResultV2Typical {
|
||||||
// TODO: use proper `Route` type from `routing-utils` (perhaps move types to a common package)
|
// TODO: use proper `Route` type from `routing-utils` (perhaps move types to a common package)
|
||||||
routes: any[];
|
routes?: any[];
|
||||||
images?: Images;
|
images?: Images;
|
||||||
output: {
|
output: {
|
||||||
[key: string]: File | Lambda;
|
[key: string]: File | Lambda | Prerender | EdgeFunction;
|
||||||
};
|
};
|
||||||
wildcard?: Array<{
|
wildcard?: Array<{
|
||||||
domain: string;
|
domain: string;
|
||||||
@@ -405,6 +392,8 @@ export interface BuildResultV2 {
|
|||||||
}>;
|
}>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type BuildResultV2 = BuildResultV2Typical | BuildResultBuildOutput;
|
||||||
|
|
||||||
export interface BuildResultV3 {
|
export interface BuildResultV3 {
|
||||||
output: Lambda;
|
output: Lambda;
|
||||||
}
|
}
|
||||||
|
|||||||
4
packages/build-utils/test/fixtures/02-zero-config-api/yarn.lock
vendored
Normal file
4
packages/build-utils/test/fixtures/02-zero-config-api/yarn.lock
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
|
||||||
|
# yarn lockfile v1
|
||||||
|
|
||||||
|
|
||||||
15
packages/build-utils/test/fixtures/22-pnpm/package.json
vendored
Normal file
15
packages/build-utils/test/fixtures/22-pnpm/package.json
vendored
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"name": "22-pnpm",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "",
|
||||||
|
"main": "index.js",
|
||||||
|
"scripts": {
|
||||||
|
"build": "mkdir -p public && (printf \"pnpm version: \" && pnpm -v) > public/index.txt"
|
||||||
|
},
|
||||||
|
"keywords": [],
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"once": "^1.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
19
packages/build-utils/test/fixtures/22-pnpm/pnpm-lock.yaml
generated
vendored
Normal file
19
packages/build-utils/test/fixtures/22-pnpm/pnpm-lock.yaml
generated
vendored
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
lockfileVersion: 5.3
|
||||||
|
|
||||||
|
specifiers:
|
||||||
|
once: ^1.4.0
|
||||||
|
|
||||||
|
dependencies:
|
||||||
|
once: 1.4.0
|
||||||
|
|
||||||
|
packages:
|
||||||
|
|
||||||
|
/once/1.4.0:
|
||||||
|
resolution: {integrity: sha1-WDsap3WWHUsROsF9nFC6753Xa9E=}
|
||||||
|
dependencies:
|
||||||
|
wrappy: 1.0.2
|
||||||
|
dev: false
|
||||||
|
|
||||||
|
/wrappy/1.0.2:
|
||||||
|
resolution: {integrity: sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=}
|
||||||
|
dev: false
|
||||||
11
packages/build-utils/test/fixtures/22-pnpm/vercel.json
vendored
Normal file
11
packages/build-utils/test/fixtures/22-pnpm/vercel.json
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"version": 2,
|
||||||
|
"builds": [{ "src": "package.json", "use": "@vercel/static-build" }],
|
||||||
|
"probes": [
|
||||||
|
{
|
||||||
|
"path": "/",
|
||||||
|
"mustContain": "pnpm version: 6",
|
||||||
|
"logMustContain": "pnpm run build"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
5
packages/build-utils/test/fixtures/23-pnpm-workspaces/c/package.json
vendored
Normal file
5
packages/build-utils/test/fixtures/23-pnpm-workspaces/c/package.json
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"name": "c",
|
||||||
|
"license": "MIT",
|
||||||
|
"version": "0.1.0"
|
||||||
|
}
|
||||||
8
packages/build-utils/test/fixtures/23-pnpm-workspaces/d/package.json
vendored
Normal file
8
packages/build-utils/test/fixtures/23-pnpm-workspaces/d/package.json
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"name": "d",
|
||||||
|
"license": "MIT",
|
||||||
|
"version": "0.1.0",
|
||||||
|
"devDependencies": {
|
||||||
|
"once": "1.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
6
packages/build-utils/test/fixtures/23-pnpm-workspaces/package.json
vendored
Normal file
6
packages/build-utils/test/fixtures/23-pnpm-workspaces/package.json
vendored
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"private": true,
|
||||||
|
"name": "23-pnpm-workspaces",
|
||||||
|
"license": "MIT",
|
||||||
|
"version": "1.0.0"
|
||||||
|
}
|
||||||
27
packages/build-utils/test/fixtures/23-pnpm-workspaces/pnpm-lock.yaml
generated
vendored
Normal file
27
packages/build-utils/test/fixtures/23-pnpm-workspaces/pnpm-lock.yaml
generated
vendored
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
lockfileVersion: 5.3
|
||||||
|
|
||||||
|
importers:
|
||||||
|
|
||||||
|
.:
|
||||||
|
specifiers: {}
|
||||||
|
|
||||||
|
c:
|
||||||
|
specifiers: {}
|
||||||
|
|
||||||
|
d:
|
||||||
|
specifiers:
|
||||||
|
once: 1.4.0
|
||||||
|
devDependencies:
|
||||||
|
once: 1.4.0
|
||||||
|
|
||||||
|
packages:
|
||||||
|
|
||||||
|
/once/1.4.0:
|
||||||
|
resolution: {integrity: sha1-WDsap3WWHUsROsF9nFC6753Xa9E=}
|
||||||
|
dependencies:
|
||||||
|
wrappy: 1.0.2
|
||||||
|
dev: true
|
||||||
|
|
||||||
|
/wrappy/1.0.2:
|
||||||
|
resolution: {integrity: sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=}
|
||||||
|
dev: true
|
||||||
3
packages/build-utils/test/fixtures/23-pnpm-workspaces/pnpm-workspace.yaml
vendored
Normal file
3
packages/build-utils/test/fixtures/23-pnpm-workspaces/pnpm-workspace.yaml
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
packages:
|
||||||
|
- 'c'
|
||||||
|
- 'd'
|
||||||
1
packages/build-utils/test/fixtures/24-pnpm-hoisted/.gitignore
vendored
Normal file
1
packages/build-utils/test/fixtures/24-pnpm-hoisted/.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
.vercel
|
||||||
1
packages/build-utils/test/fixtures/24-pnpm-hoisted/.npmrc
vendored
Normal file
1
packages/build-utils/test/fixtures/24-pnpm-hoisted/.npmrc
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
node-linker=hoisted
|
||||||
5
packages/build-utils/test/fixtures/24-pnpm-hoisted/a/index.js
vendored
Normal file
5
packages/build-utils/test/fixtures/24-pnpm-hoisted/a/index.js
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
const once = require('once');
|
||||||
|
|
||||||
|
module.exports = () => {
|
||||||
|
once(() => {});
|
||||||
|
};
|
||||||
5
packages/build-utils/test/fixtures/24-pnpm-hoisted/b/index.js
vendored
Normal file
5
packages/build-utils/test/fixtures/24-pnpm-hoisted/b/index.js
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
const once = require('once');
|
||||||
|
|
||||||
|
module.exports = () => {
|
||||||
|
once(() => {});
|
||||||
|
};
|
||||||
10
packages/build-utils/test/fixtures/24-pnpm-hoisted/index.js
vendored
Normal file
10
packages/build-utils/test/fixtures/24-pnpm-hoisted/index.js
vendored
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
const { exec } = require('exeggcute');
|
||||||
|
|
||||||
|
const a = require('./a');
|
||||||
|
const b = require('./b');
|
||||||
|
|
||||||
|
a();
|
||||||
|
b();
|
||||||
|
|
||||||
|
exec('mkdir public', __dirname);
|
||||||
|
exec('echo "Hello, World!" > public/index.html', __dirname);
|
||||||
11
packages/build-utils/test/fixtures/24-pnpm-hoisted/package.json
vendored
Normal file
11
packages/build-utils/test/fixtures/24-pnpm-hoisted/package.json
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"private": "true",
|
||||||
|
"name": "24-pnpm-hoisted",
|
||||||
|
"scripts": {
|
||||||
|
"build": "ls -Al node_modules && node index.js"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"exeggcute": "^1.0.0",
|
||||||
|
"once": "^1.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
25
packages/build-utils/test/fixtures/24-pnpm-hoisted/pnpm-lock.yaml
generated
vendored
Normal file
25
packages/build-utils/test/fixtures/24-pnpm-hoisted/pnpm-lock.yaml
generated
vendored
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
lockfileVersion: 5.3
|
||||||
|
|
||||||
|
specifiers:
|
||||||
|
exeggcute: ^1.0.0
|
||||||
|
once: ^1.4.0
|
||||||
|
|
||||||
|
dependencies:
|
||||||
|
exeggcute: 1.0.0
|
||||||
|
once: 1.4.0
|
||||||
|
|
||||||
|
packages:
|
||||||
|
|
||||||
|
/exeggcute/1.0.0:
|
||||||
|
resolution: {integrity: sha1-qLXakIhjGCDm9ggx4CvEc5mnbBU=}
|
||||||
|
dev: false
|
||||||
|
|
||||||
|
/once/1.4.0:
|
||||||
|
resolution: {integrity: sha1-WDsap3WWHUsROsF9nFC6753Xa9E=}
|
||||||
|
dependencies:
|
||||||
|
wrappy: 1.0.2
|
||||||
|
dev: false
|
||||||
|
|
||||||
|
/wrappy/1.0.2:
|
||||||
|
resolution: {integrity: sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=}
|
||||||
|
dev: false
|
||||||
20
packages/build-utils/test/fixtures/24-pnpm-hoisted/vercel.json
vendored
Normal file
20
packages/build-utils/test/fixtures/24-pnpm-hoisted/vercel.json
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"probes": [
|
||||||
|
{
|
||||||
|
"path": "/",
|
||||||
|
"mustContain": "Hello, World!"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/",
|
||||||
|
"logMustContain": "once"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/",
|
||||||
|
"logMustContain": "exeggcute"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "/",
|
||||||
|
"logMustContain": "wrappy"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -31,6 +31,7 @@ const skipFixtures: string[] = [
|
|||||||
'07-zero-config-jekyll',
|
'07-zero-config-jekyll',
|
||||||
'08-zero-config-middleman',
|
'08-zero-config-middleman',
|
||||||
'21-npm-workspaces',
|
'21-npm-workspaces',
|
||||||
|
'23-pnpm-workspaces',
|
||||||
];
|
];
|
||||||
|
|
||||||
// eslint-disable-next-line no-restricted-syntax
|
// eslint-disable-next-line no-restricted-syntax
|
||||||
|
|||||||
134
packages/build-utils/test/unit.test.ts
vendored
134
packages/build-utils/test/unit.test.ts
vendored
@@ -1,7 +1,7 @@
|
|||||||
import ms from 'ms';
|
import ms from 'ms';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import fs from 'fs-extra';
|
import fs, { readlink } from 'fs-extra';
|
||||||
import { strict as assert } from 'assert';
|
import { strict as assert, strictEqual } from 'assert';
|
||||||
import { createZip } from '../src/lambda';
|
import { createZip } from '../src/lambda';
|
||||||
import { getSupportedNodeVersion } from '../src/fs/node-version';
|
import { getSupportedNodeVersion } from '../src/fs/node-version';
|
||||||
import download from '../src/fs/download';
|
import download from '../src/fs/download';
|
||||||
@@ -14,6 +14,8 @@ import {
|
|||||||
runNpmInstall,
|
runNpmInstall,
|
||||||
runPackageJsonScript,
|
runPackageJsonScript,
|
||||||
scanParentDirs,
|
scanParentDirs,
|
||||||
|
FileBlob,
|
||||||
|
Meta,
|
||||||
} from '../src';
|
} from '../src';
|
||||||
|
|
||||||
async function expectBuilderError(promise: Promise<any>, pattern: string) {
|
async function expectBuilderError(promise: Promise<any>, pattern: string) {
|
||||||
@@ -47,7 +49,7 @@ afterEach(() => {
|
|||||||
console.warn = originalConsoleWarn;
|
console.warn = originalConsoleWarn;
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should re-create symlinks properly', async () => {
|
it('should re-create FileFsRef symlinks properly', async () => {
|
||||||
if (process.platform === 'win32') {
|
if (process.platform === 'win32') {
|
||||||
console.log('Skipping test on windows');
|
console.log('Skipping test on windows');
|
||||||
return;
|
return;
|
||||||
@@ -69,6 +71,72 @@ it('should re-create symlinks properly', async () => {
|
|||||||
assert(linkStat.isSymbolicLink());
|
assert(linkStat.isSymbolicLink());
|
||||||
assert(linkDirStat.isSymbolicLink());
|
assert(linkDirStat.isSymbolicLink());
|
||||||
assert(aStat.isFile());
|
assert(aStat.isFile());
|
||||||
|
|
||||||
|
const [linkDirContents, linkTextContents] = await Promise.all([
|
||||||
|
readlink(path.join(outDir, 'link-dir')),
|
||||||
|
readlink(path.join(outDir, 'link.txt')),
|
||||||
|
]);
|
||||||
|
|
||||||
|
strictEqual(linkDirContents, 'dir');
|
||||||
|
strictEqual(linkTextContents, './a.txt');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should re-create FileBlob symlinks properly', async () => {
|
||||||
|
if (process.platform === 'win32') {
|
||||||
|
console.log('Skipping test on windows');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const files = {
|
||||||
|
'a.txt': new FileBlob({
|
||||||
|
mode: 33188,
|
||||||
|
contentType: undefined,
|
||||||
|
data: 'a text',
|
||||||
|
}),
|
||||||
|
'dir/b.txt': new FileBlob({
|
||||||
|
mode: 33188,
|
||||||
|
contentType: undefined,
|
||||||
|
data: 'b text',
|
||||||
|
}),
|
||||||
|
'link-dir': new FileBlob({
|
||||||
|
mode: 41453,
|
||||||
|
contentType: undefined,
|
||||||
|
data: 'dir',
|
||||||
|
}),
|
||||||
|
'link.txt': new FileBlob({
|
||||||
|
mode: 41453,
|
||||||
|
contentType: undefined,
|
||||||
|
data: 'a.txt',
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
strictEqual(Object.keys(files).length, 4);
|
||||||
|
|
||||||
|
const outDir = path.join(__dirname, 'symlinks-out');
|
||||||
|
await fs.remove(outDir);
|
||||||
|
|
||||||
|
const files2 = await download(files, outDir);
|
||||||
|
strictEqual(Object.keys(files2).length, 4);
|
||||||
|
|
||||||
|
const [linkStat, linkDirStat, aStat, dirStat] = await Promise.all([
|
||||||
|
fs.lstat(path.join(outDir, 'link.txt')),
|
||||||
|
fs.lstat(path.join(outDir, 'link-dir')),
|
||||||
|
fs.lstat(path.join(outDir, 'a.txt')),
|
||||||
|
fs.lstat(path.join(outDir, 'dir')),
|
||||||
|
]);
|
||||||
|
|
||||||
|
assert(linkStat.isSymbolicLink());
|
||||||
|
assert(linkDirStat.isSymbolicLink());
|
||||||
|
assert(aStat.isFile());
|
||||||
|
assert(dirStat.isDirectory());
|
||||||
|
|
||||||
|
const [linkDirContents, linkTextContents] = await Promise.all([
|
||||||
|
readlink(path.join(outDir, 'link-dir')),
|
||||||
|
readlink(path.join(outDir, 'link.txt')),
|
||||||
|
]);
|
||||||
|
|
||||||
|
strictEqual(linkDirContents, 'dir');
|
||||||
|
strictEqual(linkTextContents, 'a.txt');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should create zip files with symlinks properly', async () => {
|
it('should create zip files with symlinks properly', async () => {
|
||||||
@@ -209,6 +277,16 @@ it('should not warn when package.json engines matches project setting from confi
|
|||||||
expect(warningMessages).toStrictEqual([]);
|
expect(warningMessages).toStrictEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should select nodejs16.x with ENABLE_EXPERIMENTAL_NODE16', async () => {
|
||||||
|
process.env.ENABLE_EXPERIMENTAL_NODE16 = '1';
|
||||||
|
const result = await getNodeVersion('/tmp', undefined, {}, {});
|
||||||
|
delete process.env.ENABLE_EXPERIMENTAL_NODE16;
|
||||||
|
expect(result).toEqual({ major: 16, range: '16.x', runtime: 'nodejs16.x' });
|
||||||
|
expect(warningMessages).toStrictEqual([
|
||||||
|
'Warning: Using experimental Node.js 16.x due to ENABLE_EXPERIMENTAL_NODE16=1',
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
it('should get latest node version', async () => {
|
it('should get latest node version', async () => {
|
||||||
expect(getLatestNodeVersion()).toHaveProperty('major', 14);
|
expect(getLatestNodeVersion()).toHaveProperty('major', 14);
|
||||||
});
|
});
|
||||||
@@ -332,3 +410,53 @@ it('should detect npm Workspaces', async () => {
|
|||||||
expect(result.cliType).toEqual('npm');
|
expect(result.cliType).toEqual('npm');
|
||||||
expect(result.lockfileVersion).toEqual(2);
|
expect(result.lockfileVersion).toEqual(2);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should detect pnpm', async () => {
|
||||||
|
const fixture = path.join(__dirname, 'fixtures', '22-pnpm');
|
||||||
|
const result = await scanParentDirs(fixture);
|
||||||
|
expect(result.cliType).toEqual('pnpm');
|
||||||
|
expect(result.lockfileVersion).toEqual(5.3);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should detect pnpm Workspaces', async () => {
|
||||||
|
const fixture = path.join(__dirname, 'fixtures', '23-pnpm-workspaces/a');
|
||||||
|
const result = await scanParentDirs(fixture);
|
||||||
|
expect(result.cliType).toEqual('pnpm');
|
||||||
|
expect(result.lockfileVersion).toEqual(5.3);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should only invoke `runNpmInstall()` once per `package.json` file (serial)', async () => {
|
||||||
|
const meta: Meta = {};
|
||||||
|
const fixture = path.join(__dirname, 'fixtures', '02-zero-config-api');
|
||||||
|
const apiDir = path.join(fixture, 'api');
|
||||||
|
const run1 = await runNpmInstall(apiDir, [], undefined, meta);
|
||||||
|
expect(run1).toEqual(true);
|
||||||
|
expect(
|
||||||
|
(meta.runNpmInstallSet as Set<string>).has(
|
||||||
|
path.join(fixture, 'package.json')
|
||||||
|
)
|
||||||
|
).toEqual(true);
|
||||||
|
const run2 = await runNpmInstall(apiDir, [], undefined, meta);
|
||||||
|
expect(run2).toEqual(false);
|
||||||
|
const run3 = await runNpmInstall(fixture, [], undefined, meta);
|
||||||
|
expect(run3).toEqual(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should only invoke `runNpmInstall()` once per `package.json` file (parallel)', async () => {
|
||||||
|
const meta: Meta = {};
|
||||||
|
const fixture = path.join(__dirname, 'fixtures', '02-zero-config-api');
|
||||||
|
const apiDir = path.join(fixture, 'api');
|
||||||
|
const [run1, run2, run3] = await Promise.all([
|
||||||
|
runNpmInstall(apiDir, [], undefined, meta),
|
||||||
|
runNpmInstall(apiDir, [], undefined, meta),
|
||||||
|
runNpmInstall(fixture, [], undefined, meta),
|
||||||
|
]);
|
||||||
|
expect(run1).toEqual(true);
|
||||||
|
expect(run2).toEqual(false);
|
||||||
|
expect(run3).toEqual(false);
|
||||||
|
expect(
|
||||||
|
(meta.runNpmInstallSet as Set<string>).has(
|
||||||
|
path.join(fixture, 'package.json')
|
||||||
|
)
|
||||||
|
).toEqual(true);
|
||||||
|
});
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "vercel",
|
"name": "vercel",
|
||||||
"version": "24.0.1-canary.1",
|
"version": "24.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",
|
||||||
@@ -43,14 +43,15 @@
|
|||||||
"node": ">= 12"
|
"node": ">= 12"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@vercel/build-utils": "2.14.1-canary.1",
|
"@vercel/build-utils": "2.16.0",
|
||||||
"@vercel/go": "1.3.1-canary.1",
|
"@vercel/go": "1.4.0",
|
||||||
"@vercel/node": "1.13.1-canary.1",
|
"@vercel/node": "1.15.0",
|
||||||
"@vercel/python": "2.2.1-canary.1",
|
"@vercel/python": "2.3.0",
|
||||||
"@vercel/ruby": "1.3.1-canary.1",
|
"@vercel/ruby": "1.3.3",
|
||||||
"update-notifier": "4.1.0"
|
"update-notifier": "4.1.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@alex_neo/jest-expect-message": "1.0.5",
|
||||||
"@next/env": "11.1.2",
|
"@next/env": "11.1.2",
|
||||||
"@sentry/node": "5.5.0",
|
"@sentry/node": "5.5.0",
|
||||||
"@sindresorhus/slugify": "0.11.0",
|
"@sindresorhus/slugify": "0.11.0",
|
||||||
@@ -68,7 +69,8 @@
|
|||||||
"@types/glob": "7.1.1",
|
"@types/glob": "7.1.1",
|
||||||
"@types/http-proxy": "1.16.2",
|
"@types/http-proxy": "1.16.2",
|
||||||
"@types/inquirer": "7.3.1",
|
"@types/inquirer": "7.3.1",
|
||||||
"@types/jest": "27.0.1",
|
"@types/jest": "27.4.1",
|
||||||
|
"@types/jest-expect-message": "1.0.3",
|
||||||
"@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",
|
||||||
@@ -88,11 +90,10 @@
|
|||||||
"@types/update-notifier": "5.1.0",
|
"@types/update-notifier": "5.1.0",
|
||||||
"@types/which": "1.3.2",
|
"@types/which": "1.3.2",
|
||||||
"@types/write-json-file": "2.2.1",
|
"@types/write-json-file": "2.2.1",
|
||||||
"@vercel/client": "10.3.1-canary.1",
|
"@vercel/client": "11.0.0",
|
||||||
"@vercel/fetch-retry": "5.0.3",
|
"@vercel/fetch-retry": "5.0.3",
|
||||||
"@vercel/frameworks": "0.6.1-canary.1",
|
"@vercel/frameworks": "0.8.0",
|
||||||
"@vercel/ncc": "0.24.0",
|
"@vercel/ncc": "0.24.0",
|
||||||
"@vercel/nft": "0.17.5",
|
|
||||||
"@zeit/fun": "0.11.2",
|
"@zeit/fun": "0.11.2",
|
||||||
"@zeit/source-map-support": "0.6.2",
|
"@zeit/source-map-support": "0.6.2",
|
||||||
"ajv": "6.12.2",
|
"ajv": "6.12.2",
|
||||||
@@ -161,7 +162,6 @@
|
|||||||
"title": "3.4.1",
|
"title": "3.4.1",
|
||||||
"tmp-promise": "1.0.3",
|
"tmp-promise": "1.0.3",
|
||||||
"tree-kill": "1.2.2",
|
"tree-kill": "1.2.2",
|
||||||
"ts-eager": "2.0.2",
|
|
||||||
"ts-node": "8.3.0",
|
"ts-node": "8.3.0",
|
||||||
"typescript": "4.3.4",
|
"typescript": "4.3.4",
|
||||||
"universal-analytics": "0.4.20",
|
"universal-analytics": "0.4.20",
|
||||||
@@ -178,6 +178,9 @@
|
|||||||
"isolatedModules": true
|
"isolatedModules": true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"setupFilesAfterEnv": [
|
||||||
|
"@alex_neo/jest-expect-message"
|
||||||
|
],
|
||||||
"verbose": false,
|
"verbose": false,
|
||||||
"testEnvironment": "node",
|
"testEnvironment": "node",
|
||||||
"testMatch": [
|
"testMatch": [
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ export default async function set(
|
|||||||
if (args.length === 0) {
|
if (args.length === 0) {
|
||||||
output.error(
|
output.error(
|
||||||
`To ship to production, optionally configure your domains (${link(
|
`To ship to production, optionally configure your domains (${link(
|
||||||
'https://vercel.com/docs/v2/custom-domains'
|
'https://vercel.link/domain-configuration'
|
||||||
)}) and run ${getCommandName(`--prod`)}.`
|
)}) and run ${getCommandName(`--prod`)}.`
|
||||||
);
|
);
|
||||||
return 1;
|
return 1;
|
||||||
|
|||||||
@@ -42,6 +42,7 @@ export const help = () => `
|
|||||||
|
|
||||||
-h, --help Output usage information
|
-h, --help Output usage information
|
||||||
-v, --version Output the version number
|
-v, --version Output the version number
|
||||||
|
--cwd Current working directory
|
||||||
-V, --platform-version Set the platform version to deploy to
|
-V, --platform-version Set the platform version to deploy to
|
||||||
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
-A ${chalk.bold.underline('FILE')}, --local-config=${chalk.bold.underline(
|
||||||
'FILE'
|
'FILE'
|
||||||
|
|||||||
@@ -181,6 +181,21 @@ export default async (client: Client) => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// build `--prebuilt`
|
||||||
|
if (argv['--prebuilt']) {
|
||||||
|
const prebuiltExists = await fs.pathExists(join(path, '.vercel/output'));
|
||||||
|
if (!prebuiltExists) {
|
||||||
|
error(
|
||||||
|
`The ${param(
|
||||||
|
'--prebuilt'
|
||||||
|
)} option was used, but no prebuilt output found in ".vercel/output". Run ${getCommandName(
|
||||||
|
'build'
|
||||||
|
)} to generate a local build.`
|
||||||
|
);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// retrieve `project` and `org` from .vercel
|
// retrieve `project` and `org` from .vercel
|
||||||
const link = await getLinkedProject(client, path);
|
const link = await getLinkedProject(client, path);
|
||||||
|
|
||||||
@@ -892,8 +907,8 @@ const parseEnv = (env?: string[] | Dictionary<string>) => {
|
|||||||
if (equalsSign === -1) {
|
if (equalsSign === -1) {
|
||||||
key = e;
|
key = e;
|
||||||
} else {
|
} else {
|
||||||
key = e.substr(0, equalsSign);
|
key = e.slice(0, equalsSign);
|
||||||
value = e.substr(equalsSign + 1);
|
value = e.slice(equalsSign + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
o[key] = value;
|
o[key] = value;
|
||||||
|
|||||||
22
packages/cli/src/commands/env/index.ts
vendored
22
packages/cli/src/commands/env/index.ts
vendored
@@ -1,4 +1,5 @@
|
|||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
|
import { ProjectEnvTarget } from '../../types';
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
import { getEnvTargetPlaceholder } from '../../util/env/env-target';
|
import { getEnvTargetPlaceholder } from '../../util/env/env-target';
|
||||||
import getArgs from '../../util/get-args';
|
import getArgs from '../../util/get-args';
|
||||||
@@ -114,9 +115,11 @@ export default async function main(client: Client) {
|
|||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { subcommand, args } = getSubcommand(argv._.slice(1), COMMAND_CONFIG);
|
const cwd = argv['--cwd'] || process.cwd();
|
||||||
|
const subArgs = argv._.slice(1);
|
||||||
|
const { subcommand, args } = getSubcommand(subArgs, COMMAND_CONFIG);
|
||||||
const { output, config } = client;
|
const { output, config } = client;
|
||||||
const link = await getLinkedProject(client);
|
const link = await getLinkedProject(client, cwd);
|
||||||
if (link.status === 'error') {
|
if (link.status === 'error') {
|
||||||
return link.exitCode;
|
return link.exitCode;
|
||||||
} else if (link.status === 'not_linked') {
|
} else if (link.status === 'not_linked') {
|
||||||
@@ -137,14 +140,15 @@ export default async function main(client: Client) {
|
|||||||
case 'rm':
|
case 'rm':
|
||||||
return rm(client, project, argv, args, output);
|
return rm(client, project, argv, args, output);
|
||||||
case 'pull':
|
case 'pull':
|
||||||
output.warn(
|
return pull(
|
||||||
`${getCommandName(
|
client,
|
||||||
'env pull'
|
project,
|
||||||
)} is deprecated and will be removed in future releases. Run ${getCommandName(
|
ProjectEnvTarget.Development,
|
||||||
'pull'
|
argv,
|
||||||
)} instead.`
|
args,
|
||||||
|
output,
|
||||||
|
cwd
|
||||||
);
|
);
|
||||||
return pull(client, project, argv, args, output);
|
|
||||||
default:
|
default:
|
||||||
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
output.error(getInvalidSubcommand(COMMAND_CONFIG));
|
||||||
help();
|
help();
|
||||||
|
|||||||
22
packages/cli/src/commands/env/pull.ts
vendored
22
packages/cli/src/commands/env/pull.ts
vendored
@@ -1,7 +1,8 @@
|
|||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
import { closeSync, openSync, promises, readSync } from 'fs';
|
import { outputFile } from 'fs-extra';
|
||||||
|
import { closeSync, openSync, readSync } from 'fs';
|
||||||
import { resolve } from 'path';
|
import { resolve } from 'path';
|
||||||
import { Project } from '../../types';
|
import { Project, ProjectEnvTarget } from '../../types';
|
||||||
import Client from '../../util/client';
|
import Client from '../../util/client';
|
||||||
import exposeSystemEnvs from '../../util/dev/expose-system-envs';
|
import exposeSystemEnvs from '../../util/dev/expose-system-envs';
|
||||||
import { emoji, prependEmoji } from '../../util/emoji';
|
import { emoji, prependEmoji } from '../../util/emoji';
|
||||||
@@ -12,7 +13,6 @@ import { Output } from '../../util/output';
|
|||||||
import param from '../../util/output/param';
|
import param from '../../util/output/param';
|
||||||
import stamp from '../../util/output/stamp';
|
import stamp from '../../util/output/stamp';
|
||||||
import { getCommandName } from '../../util/pkg-name';
|
import { getCommandName } from '../../util/pkg-name';
|
||||||
const { writeFile } = promises;
|
|
||||||
|
|
||||||
const CONTENTS_PREFIX = '# Created by Vercel CLI\n';
|
const CONTENTS_PREFIX = '# Created by Vercel CLI\n';
|
||||||
|
|
||||||
@@ -45,9 +45,11 @@ function tryReadHeadSync(path: string, length: number) {
|
|||||||
export default async function pull(
|
export default async function pull(
|
||||||
client: Client,
|
client: Client,
|
||||||
project: Project,
|
project: Project,
|
||||||
|
environment: ProjectEnvTarget,
|
||||||
opts: Partial<Options>,
|
opts: Partial<Options>,
|
||||||
args: string[],
|
args: string[],
|
||||||
output: Output
|
output: Output,
|
||||||
|
cwd: string
|
||||||
) {
|
) {
|
||||||
if (args.length > 1) {
|
if (args.length > 1) {
|
||||||
output.error(
|
output.error(
|
||||||
@@ -58,7 +60,7 @@ export default async function pull(
|
|||||||
|
|
||||||
// handle relative or absolute filename
|
// handle relative or absolute filename
|
||||||
const [filename = '.env'] = args;
|
const [filename = '.env'] = args;
|
||||||
const fullPath = resolve(filename);
|
const fullPath = resolve(cwd, filename);
|
||||||
const skipConfirmation = opts['--yes'];
|
const skipConfirmation = opts['--yes'];
|
||||||
|
|
||||||
const head = tryReadHeadSync(fullPath, Buffer.byteLength(CONTENTS_PREFIX));
|
const head = tryReadHeadSync(fullPath, Buffer.byteLength(CONTENTS_PREFIX));
|
||||||
@@ -79,7 +81,7 @@ export default async function pull(
|
|||||||
}
|
}
|
||||||
|
|
||||||
output.print(
|
output.print(
|
||||||
`Downloading Development Environment Variables for Project ${chalk.bold(
|
`Downloading "${environment}" Environment Variables for Project ${chalk.bold(
|
||||||
project.name
|
project.name
|
||||||
)}\n`
|
)}\n`
|
||||||
);
|
);
|
||||||
@@ -88,7 +90,7 @@ export default async function pull(
|
|||||||
output.spinner('Downloading');
|
output.spinner('Downloading');
|
||||||
|
|
||||||
const [{ envs: projectEnvs }, { systemEnvValues }] = await Promise.all([
|
const [{ envs: projectEnvs }, { systemEnvValues }] = await Promise.all([
|
||||||
getDecryptedEnvRecords(output, client, project.id),
|
getDecryptedEnvRecords(output, client, project.id, environment),
|
||||||
project.autoExposeSystemEnvs
|
project.autoExposeSystemEnvs
|
||||||
? getSystemEnvValues(output, client, project.id)
|
? getSystemEnvValues(output, client, project.id)
|
||||||
: { systemEnvValues: [] },
|
: { systemEnvValues: [] },
|
||||||
@@ -97,7 +99,9 @@ export default async function pull(
|
|||||||
const records = exposeSystemEnvs(
|
const records = exposeSystemEnvs(
|
||||||
projectEnvs,
|
projectEnvs,
|
||||||
systemEnvValues,
|
systemEnvValues,
|
||||||
project.autoExposeSystemEnvs
|
project.autoExposeSystemEnvs,
|
||||||
|
undefined,
|
||||||
|
environment
|
||||||
);
|
);
|
||||||
|
|
||||||
const contents =
|
const contents =
|
||||||
@@ -107,7 +111,7 @@ export default async function pull(
|
|||||||
.join('\n') +
|
.join('\n') +
|
||||||
'\n';
|
'\n';
|
||||||
|
|
||||||
await writeFile(fullPath, contents, 'utf8');
|
await outputFile(fullPath, contents, 'utf8');
|
||||||
|
|
||||||
output.print(
|
output.print(
|
||||||
`${prependEmoji(
|
`${prependEmoji(
|
||||||
|
|||||||
@@ -83,6 +83,7 @@ async function fetchExampleList(client: Client) {
|
|||||||
const url = `${EXAMPLE_API}/v2/list.json`;
|
const url = `${EXAMPLE_API}/v2/list.json`;
|
||||||
|
|
||||||
const body = await client.fetch<Example[]>(url);
|
const body = await client.fetch<Example[]>(url);
|
||||||
|
client.output.stopSpinner();
|
||||||
return body;
|
return body;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
import { join } from 'path';
|
import { join } from 'path';
|
||||||
import Client from '../util/client';
|
import Client from '../util/client';
|
||||||
|
import { ProjectEnvTarget } from '../types';
|
||||||
import { emoji, prependEmoji } from '../util/emoji';
|
import { emoji, prependEmoji } from '../util/emoji';
|
||||||
import getArgs from '../util/get-args';
|
import getArgs from '../util/get-args';
|
||||||
import handleError from '../util/handle-error';
|
|
||||||
import setupAndLink from '../util/link/setup-and-link';
|
import setupAndLink from '../util/link/setup-and-link';
|
||||||
import logo from '../util/output/logo';
|
import logo from '../util/output/logo';
|
||||||
import stamp from '../util/output/stamp';
|
import stamp from '../util/output/stamp';
|
||||||
@@ -14,7 +14,13 @@ import {
|
|||||||
VERCEL_DIR_PROJECT,
|
VERCEL_DIR_PROJECT,
|
||||||
} from '../util/projects/link';
|
} from '../util/projects/link';
|
||||||
import { writeProjectSettings } from '../util/projects/project-settings';
|
import { writeProjectSettings } from '../util/projects/project-settings';
|
||||||
import pull from './env/pull';
|
import envPull from './env/pull';
|
||||||
|
|
||||||
|
import type { Project, Org } from '../types';
|
||||||
|
import {
|
||||||
|
isValidEnvTarget,
|
||||||
|
getEnvTargetPlaceholder,
|
||||||
|
} from '../util/env/env-target';
|
||||||
|
|
||||||
const help = () => {
|
const help = () => {
|
||||||
return console.log(`
|
return console.log(`
|
||||||
@@ -30,7 +36,7 @@ const help = () => {
|
|||||||
'DIR'
|
'DIR'
|
||||||
)} Path to the global ${'`.vercel`'} directory
|
)} Path to the global ${'`.vercel`'} directory
|
||||||
-d, --debug Debug mode [off]
|
-d, --debug Debug mode [off]
|
||||||
--env [filename] The file to write Development Environment Variables to [.env]
|
--environment [environment] Deployment environment [development]
|
||||||
-y, --yes Skip the confirmation prompt
|
-y, --yes Skip the confirmation prompt
|
||||||
|
|
||||||
${chalk.dim('Examples:')}
|
${chalk.dim('Examples:')}
|
||||||
@@ -41,32 +47,46 @@ const help = () => {
|
|||||||
${chalk.cyan(`$ ${getPkgName()} pull ./path-to-project`)}
|
${chalk.cyan(`$ ${getPkgName()} pull ./path-to-project`)}
|
||||||
${chalk.cyan(`$ ${getPkgName()} pull --env .env.local`)}
|
${chalk.cyan(`$ ${getPkgName()} pull --env .env.local`)}
|
||||||
${chalk.cyan(`$ ${getPkgName()} pull ./path-to-project --env .env.local`)}
|
${chalk.cyan(`$ ${getPkgName()} pull ./path-to-project --env .env.local`)}
|
||||||
|
|
||||||
|
${chalk.gray('–')} Pull specific environment's Project Settings from the cloud
|
||||||
|
|
||||||
|
${chalk.cyan(
|
||||||
|
`$ ${getPkgName()} pull --environment=${getEnvTargetPlaceholder()}`
|
||||||
|
)}
|
||||||
`);
|
`);
|
||||||
};
|
};
|
||||||
export default async function main(client: Client) {
|
|
||||||
let argv;
|
function processArgs(client: Client) {
|
||||||
try {
|
return getArgs(client.argv.slice(2), {
|
||||||
argv = getArgs(client.argv.slice(2), {
|
|
||||||
'--yes': Boolean,
|
'--yes': Boolean,
|
||||||
'--env': String,
|
'--env': String, // deprecated
|
||||||
|
'--environment': String,
|
||||||
'--debug': Boolean,
|
'--debug': Boolean,
|
||||||
'-d': '--debug',
|
'-d': '--debug',
|
||||||
'-y': '--yes',
|
'-y': '--yes',
|
||||||
});
|
});
|
||||||
} catch (err) {
|
|
||||||
handleError(err);
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function parseArgs(client: Client) {
|
||||||
|
const argv = processArgs(client);
|
||||||
|
|
||||||
if (argv['--help']) {
|
if (argv['--help']) {
|
||||||
help();
|
help();
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
const cwd = argv._[1] || process.cwd();
|
return argv;
|
||||||
const yes = argv['--yes'];
|
}
|
||||||
const env = argv['--env'] ?? '.env';
|
|
||||||
const settingsStamp = stamp();
|
type LinkResult = {
|
||||||
|
org: Org;
|
||||||
|
project: Project;
|
||||||
|
};
|
||||||
|
async function ensureLink(
|
||||||
|
client: Client,
|
||||||
|
cwd: string,
|
||||||
|
yes: boolean
|
||||||
|
): Promise<LinkResult | number> {
|
||||||
let link = await getLinkedProject(client, cwd);
|
let link = await getLinkedProject(client, cwd);
|
||||||
if (link.status === 'not_linked') {
|
if (link.status === 'not_linked') {
|
||||||
link = await setupAndLink(client, cwd, {
|
link = await setupAndLink(client, cwd, {
|
||||||
@@ -85,24 +105,70 @@ export default async function main(client: Client) {
|
|||||||
return link.exitCode;
|
return link.exitCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return { org: link.org, project: link.project };
|
||||||
|
}
|
||||||
|
|
||||||
|
async function pullAllEnvFiles(
|
||||||
|
environment: ProjectEnvTarget,
|
||||||
|
client: Client,
|
||||||
|
project: Project,
|
||||||
|
argv: ReturnType<typeof processArgs>,
|
||||||
|
cwd: string
|
||||||
|
): Promise<number> {
|
||||||
|
const environmentFile = `.env.${environment}.local`;
|
||||||
|
return envPull(
|
||||||
|
client,
|
||||||
|
project,
|
||||||
|
environment,
|
||||||
|
argv,
|
||||||
|
[join('.vercel', environmentFile)],
|
||||||
|
client.output,
|
||||||
|
cwd
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseEnvironment(environment = 'development'): ProjectEnvTarget {
|
||||||
|
if (!isValidEnvTarget(environment)) {
|
||||||
|
throw new Error(
|
||||||
|
`environment "${environment}" not supported; must be one of ${getEnvTargetPlaceholder()}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return environment;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function main(client: Client) {
|
||||||
|
const argv = parseArgs(client);
|
||||||
|
if (typeof argv === 'number') {
|
||||||
|
return argv;
|
||||||
|
}
|
||||||
|
|
||||||
|
const cwd = argv._[1] || process.cwd();
|
||||||
|
const yes = Boolean(argv['--yes']);
|
||||||
|
const environment = parseEnvironment(argv['--environment'] || undefined);
|
||||||
|
|
||||||
|
const link = await ensureLink(client, cwd, yes);
|
||||||
|
if (typeof link === 'number') {
|
||||||
|
return link;
|
||||||
|
}
|
||||||
|
|
||||||
const { project, org } = link;
|
const { project, org } = link;
|
||||||
|
|
||||||
client.config.currentTeam = org.type === 'team' ? org.id : undefined;
|
client.config.currentTeam = org.type === 'team' ? org.id : undefined;
|
||||||
|
|
||||||
const result = await pull(
|
const pullResultCode = await pullAllEnvFiles(
|
||||||
|
environment,
|
||||||
client,
|
client,
|
||||||
project,
|
project,
|
||||||
argv,
|
argv,
|
||||||
[join(cwd, env)],
|
cwd
|
||||||
client.output
|
|
||||||
);
|
);
|
||||||
if (result !== 0) {
|
if (pullResultCode !== 0) {
|
||||||
// an error happened
|
return pullResultCode;
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await writeProjectSettings(cwd, project, org);
|
await writeProjectSettings(cwd, project, org);
|
||||||
|
|
||||||
|
const settingsStamp = stamp();
|
||||||
client.output.print(
|
client.output.print(
|
||||||
`${prependEmoji(
|
`${prependEmoji(
|
||||||
`Downloaded project settings to ${chalk.bold(
|
`Downloaded project settings to ${chalk.bold(
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ const emailAutoComplete = (value: string, teamSlug: string) => {
|
|||||||
domains.unshift(teamSlug);
|
domains.unshift(teamSlug);
|
||||||
for (const domain of domains) {
|
for (const domain of domains) {
|
||||||
if (domain.startsWith(host)) {
|
if (domain.startsWith(host)) {
|
||||||
suggestion = domain.substr(host.length);
|
suggestion = domain.slice(host.length);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ try {
|
|||||||
}
|
}
|
||||||
|
|
||||||
import { join } from 'path';
|
import { join } from 'path';
|
||||||
import { existsSync, lstatSync } from 'fs';
|
import { existsSync } from 'fs';
|
||||||
import sourceMap from '@zeit/source-map-support';
|
import sourceMap from '@zeit/source-map-support';
|
||||||
import { mkdirp } from 'fs-extra';
|
import { mkdirp } from 'fs-extra';
|
||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
@@ -136,6 +136,11 @@ const main = async () => {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const cwd = argv['--cwd'];
|
||||||
|
if (cwd) {
|
||||||
|
process.chdir(cwd);
|
||||||
|
}
|
||||||
|
|
||||||
// Print update information, if available
|
// Print update information, if available
|
||||||
if (notifier.update && notifier.update.latest !== pkg.version && isTTY) {
|
if (notifier.update && notifier.update.latest !== pkg.version && isTTY) {
|
||||||
const { latest } = notifier.update;
|
const { latest } = notifier.update;
|
||||||
@@ -386,34 +391,11 @@ const main = async () => {
|
|||||||
GLOBAL_COMMANDS.has(targetOrSubcommand) ||
|
GLOBAL_COMMANDS.has(targetOrSubcommand) ||
|
||||||
commands.has(targetOrSubcommand);
|
commands.has(targetOrSubcommand);
|
||||||
|
|
||||||
if (targetPathExists && subcommandExists) {
|
if (targetPathExists && subcommandExists && !argv['--cwd']) {
|
||||||
const fileType = lstatSync(targetPath).isDirectory()
|
output.warn(
|
||||||
? 'subdirectory'
|
`Did you mean to deploy the subdirectory "${targetOrSubcommand}"? ` +
|
||||||
: 'file';
|
`Use \`vc --cwd ${targetOrSubcommand}\` instead.`
|
||||||
const plural = targetOrSubcommand + 's';
|
|
||||||
const singular = targetOrSubcommand.endsWith('s')
|
|
||||||
? targetOrSubcommand.slice(0, -1)
|
|
||||||
: '';
|
|
||||||
let alternative = '';
|
|
||||||
if (commands.has(plural)) {
|
|
||||||
alternative = plural;
|
|
||||||
} else if (commands.has(singular)) {
|
|
||||||
alternative = singular;
|
|
||||||
}
|
|
||||||
console.error(
|
|
||||||
error(
|
|
||||||
`The supplied argument ${param(targetOrSubcommand)} is ambiguous.` +
|
|
||||||
`\nIf you wish to deploy the ${fileType} ${param(
|
|
||||||
targetOrSubcommand
|
|
||||||
)}, first run "cd ${targetOrSubcommand}". ` +
|
|
||||||
(alternative
|
|
||||||
? `\nIf you wish to use the subcommand ${param(
|
|
||||||
targetOrSubcommand
|
|
||||||
)}, use ${param(alternative)} instead.`
|
|
||||||
: '')
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (subcommandExists) {
|
if (subcommandExists) {
|
||||||
@@ -603,73 +585,73 @@ const main = async () => {
|
|||||||
let func: any;
|
let func: any;
|
||||||
switch (targetCommand) {
|
switch (targetCommand) {
|
||||||
case 'alias':
|
case 'alias':
|
||||||
func = await import('./commands/alias');
|
func = require('./commands/alias').default;
|
||||||
break;
|
break;
|
||||||
case 'billing':
|
case 'billing':
|
||||||
func = await import('./commands/billing');
|
func = require('./commands/billing').default;
|
||||||
break;
|
break;
|
||||||
case 'bisect':
|
case 'bisect':
|
||||||
func = await import('./commands/bisect');
|
func = require('./commands/bisect').default;
|
||||||
break;
|
break;
|
||||||
case 'certs':
|
case 'certs':
|
||||||
func = await import('./commands/certs');
|
func = require('./commands/certs').default;
|
||||||
break;
|
break;
|
||||||
case 'deploy':
|
case 'deploy':
|
||||||
func = await import('./commands/deploy');
|
func = require('./commands/deploy').default;
|
||||||
break;
|
break;
|
||||||
case 'dev':
|
case 'dev':
|
||||||
func = await import('./commands/dev');
|
func = require('./commands/dev').default;
|
||||||
break;
|
break;
|
||||||
case 'dns':
|
case 'dns':
|
||||||
func = await import('./commands/dns');
|
func = require('./commands/dns').default;
|
||||||
break;
|
break;
|
||||||
case 'domains':
|
case 'domains':
|
||||||
func = await import('./commands/domains');
|
func = require('./commands/domains').default;
|
||||||
break;
|
break;
|
||||||
case 'env':
|
case 'env':
|
||||||
func = await import('./commands/env');
|
func = require('./commands/env').default;
|
||||||
break;
|
break;
|
||||||
case 'init':
|
case 'init':
|
||||||
func = await import('./commands/init');
|
func = require('./commands/init').default;
|
||||||
break;
|
break;
|
||||||
case 'inspect':
|
case 'inspect':
|
||||||
func = await import('./commands/inspect');
|
func = require('./commands/inspect').default;
|
||||||
break;
|
break;
|
||||||
case 'link':
|
case 'link':
|
||||||
func = await import('./commands/link');
|
func = require('./commands/link').default;
|
||||||
break;
|
break;
|
||||||
case 'list':
|
case 'list':
|
||||||
func = await import('./commands/list');
|
func = require('./commands/list').default;
|
||||||
break;
|
break;
|
||||||
case 'logs':
|
case 'logs':
|
||||||
func = await import('./commands/logs');
|
func = require('./commands/logs').default;
|
||||||
break;
|
break;
|
||||||
case 'login':
|
case 'login':
|
||||||
func = await import('./commands/login');
|
func = require('./commands/login').default;
|
||||||
break;
|
break;
|
||||||
case 'logout':
|
case 'logout':
|
||||||
func = await import('./commands/logout');
|
func = require('./commands/logout').default;
|
||||||
break;
|
break;
|
||||||
case 'projects':
|
case 'projects':
|
||||||
func = await import('./commands/projects');
|
func = require('./commands/projects').default;
|
||||||
break;
|
break;
|
||||||
case 'pull':
|
case 'pull':
|
||||||
func = await import('./commands/pull');
|
func = require('./commands/pull').default;
|
||||||
break;
|
break;
|
||||||
case 'remove':
|
case 'remove':
|
||||||
func = await import('./commands/remove');
|
func = require('./commands/remove').default;
|
||||||
break;
|
break;
|
||||||
case 'secrets':
|
case 'secrets':
|
||||||
func = await import('./commands/secrets');
|
func = require('./commands/secrets').default;
|
||||||
break;
|
break;
|
||||||
case 'teams':
|
case 'teams':
|
||||||
func = await import('./commands/teams');
|
func = require('./commands/teams').default;
|
||||||
break;
|
break;
|
||||||
case 'update':
|
case 'update':
|
||||||
func = await import('./commands/update');
|
func = require('./commands/update').default;
|
||||||
break;
|
break;
|
||||||
case 'whoami':
|
case 'whoami':
|
||||||
func = await import('./commands/whoami');
|
func = require('./commands/whoami').default;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
func = null;
|
func = null;
|
||||||
|
|||||||
@@ -25,7 +25,9 @@ const ARG_COMMON = {
|
|||||||
|
|
||||||
'--api': String,
|
'--api': String,
|
||||||
|
|
||||||
'--target': String
|
'--target': String,
|
||||||
|
|
||||||
|
'--cwd': String,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default () => ARG_COMMON;
|
export default () => ARG_COMMON;
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import {
|
|||||||
import { Output } from '../output';
|
import { Output } from '../output';
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import Now from '../../util';
|
import Now from '../../util';
|
||||||
import { VercelConfig } from '../dev/types';
|
|
||||||
import { Org } from '../../types';
|
import { Org } from '../../types';
|
||||||
import ua from '../ua';
|
import ua from '../ua';
|
||||||
import { linkFolderToProject } from '../projects/link';
|
import { linkFolderToProject } from '../projects/link';
|
||||||
@@ -43,7 +42,6 @@ export default async function processDeployment({
|
|||||||
uploadStamp: () => string;
|
uploadStamp: () => string;
|
||||||
deployStamp: () => string;
|
deployStamp: () => string;
|
||||||
quiet: boolean;
|
quiet: boolean;
|
||||||
nowConfig?: VercelConfig;
|
|
||||||
force?: boolean;
|
force?: boolean;
|
||||||
withCache?: boolean;
|
withCache?: boolean;
|
||||||
org: Org;
|
org: Org;
|
||||||
@@ -62,7 +60,6 @@ export default async function processDeployment({
|
|||||||
deployStamp,
|
deployStamp,
|
||||||
force,
|
force,
|
||||||
withCache,
|
withCache,
|
||||||
nowConfig,
|
|
||||||
quiet,
|
quiet,
|
||||||
prebuilt,
|
prebuilt,
|
||||||
rootDirectory,
|
rootDirectory,
|
||||||
@@ -104,11 +101,7 @@ export default async function processDeployment({
|
|||||||
const indications = [];
|
const indications = [];
|
||||||
|
|
||||||
try {
|
try {
|
||||||
for await (const event of createDeployment(
|
for await (const event of createDeployment(clientOptions, requestBody)) {
|
||||||
clientOptions,
|
|
||||||
requestBody,
|
|
||||||
nowConfig
|
|
||||||
)) {
|
|
||||||
if (['tip', 'notice', 'warning'].includes(event.type)) {
|
if (['tip', 'notice', 'warning'].includes(event.type)) {
|
||||||
indications.push(event);
|
indications.push(event);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -288,7 +288,7 @@ export async function executeBuild(
|
|||||||
// Convert the JSON-ified output map back into their corresponding `File`
|
// Convert the JSON-ified output map back into their corresponding `File`
|
||||||
// subclass type instances.
|
// subclass type instances.
|
||||||
for (const name of Object.keys(output)) {
|
for (const name of Object.keys(output)) {
|
||||||
const obj = output[name] as File;
|
const obj = output[name] as File | Lambda;
|
||||||
let lambda: BuiltLambda;
|
let lambda: BuiltLambda;
|
||||||
let fileRef: FileFsRef;
|
let fileRef: FileFsRef;
|
||||||
let fileBlob: FileBlob;
|
let fileBlob: FileBlob;
|
||||||
|
|||||||
@@ -1,4 +1,8 @@
|
|||||||
import { ProjectEnvType, ProjectEnvVariable } from '../../types';
|
import {
|
||||||
|
ProjectEnvType,
|
||||||
|
ProjectEnvVariable,
|
||||||
|
ProjectEnvTarget,
|
||||||
|
} from '../../types';
|
||||||
import { Env } from '@vercel/build-utils';
|
import { Env } from '@vercel/build-utils';
|
||||||
|
|
||||||
function getSystemEnvValue(
|
function getSystemEnvValue(
|
||||||
@@ -16,13 +20,14 @@ export default function exposeSystemEnvs(
|
|||||||
projectEnvs: ProjectEnvVariable[],
|
projectEnvs: ProjectEnvVariable[],
|
||||||
systemEnvValues: string[],
|
systemEnvValues: string[],
|
||||||
autoExposeSystemEnvs: boolean | undefined,
|
autoExposeSystemEnvs: boolean | undefined,
|
||||||
vercelUrl?: string
|
vercelUrl?: string,
|
||||||
|
target?: ProjectEnvTarget
|
||||||
) {
|
) {
|
||||||
const envs: Env = {};
|
const envs: Env = {};
|
||||||
|
|
||||||
if (autoExposeSystemEnvs) {
|
if (autoExposeSystemEnvs) {
|
||||||
envs['VERCEL'] = '1';
|
envs['VERCEL'] = '1';
|
||||||
envs['VERCEL_ENV'] = 'development';
|
envs['VERCEL_ENV'] = target || 'development';
|
||||||
|
|
||||||
for (const key of systemEnvValues) {
|
for (const key of systemEnvValues) {
|
||||||
envs[key] = getSystemEnvValue(key, { vercelUrl });
|
envs[key] = getSystemEnvValue(key, { vercelUrl });
|
||||||
|
|||||||
2
packages/cli/src/util/env/add-env-record.ts
vendored
2
packages/cli/src/util/env/add-env-record.ts
vendored
@@ -26,7 +26,7 @@ export default async function addEnvRecord(
|
|||||||
target: targets,
|
target: targets,
|
||||||
gitBranch: gitBranch || undefined,
|
gitBranch: gitBranch || undefined,
|
||||||
};
|
};
|
||||||
const url = `/v7/projects/${projectId}/env`;
|
const url = `/v8/projects/${projectId}/env`;
|
||||||
await client.fetch(url, {
|
await client.fetch(url, {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
body: JSON.stringify(body),
|
body: JSON.stringify(body),
|
||||||
|
|||||||
2
packages/cli/src/util/env/get-env-records.ts
vendored
2
packages/cli/src/util/env/get-env-records.ts
vendored
@@ -32,7 +32,7 @@ export default async function getEnvRecords(
|
|||||||
query.set('decrypt', decrypt.toString());
|
query.set('decrypt', decrypt.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
const url = `/v7/projects/${projectId}/env?${query}`;
|
const url = `/v8/projects/${projectId}/env?${query}`;
|
||||||
|
|
||||||
return client.fetch<{ envs: ProjectEnvVariable[] }>(url);
|
return client.fetch<{ envs: ProjectEnvVariable[] }>(url);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ export default async function removeEnvRecord(
|
|||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
output.debug(`Removing Environment Variable ${env.key}`);
|
output.debug(`Removing Environment Variable ${env.key}`);
|
||||||
|
|
||||||
const urlProject = `/v7/projects/${projectId}/env/${env.id}`;
|
const urlProject = `/v8/projects/${projectId}/env/${env.id}`;
|
||||||
|
|
||||||
await client.fetch<ProjectEnvVariable>(urlProject, {
|
await client.fetch<ProjectEnvVariable>(urlProject, {
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
|
|||||||
@@ -42,12 +42,15 @@ export default async function getConfig(
|
|||||||
if (localConfig instanceof CantParseJSONFile) {
|
if (localConfig instanceof CantParseJSONFile) {
|
||||||
return localConfig;
|
return localConfig;
|
||||||
}
|
}
|
||||||
if (localConfig !== null) {
|
|
||||||
|
if (localConfig === null) {
|
||||||
|
return new CantFindConfig([humanizePath(localFilePath)]);
|
||||||
|
}
|
||||||
|
|
||||||
config = localConfig;
|
config = localConfig;
|
||||||
config[fileNameSymbol] = configFile;
|
config[fileNameSymbol] = configFile;
|
||||||
return config;
|
return config;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Then try with `vercel.json` or `now.json` in the same directory
|
// Then try with `vercel.json` or `now.json` in the same directory
|
||||||
const vercelFilePath = path.resolve(localPath, 'vercel.json');
|
const vercelFilePath = path.resolve(localPath, 'vercel.json');
|
||||||
|
|||||||
@@ -11,10 +11,11 @@ import getEnvRecords from './env/get-env-records';
|
|||||||
export default async function getDecryptedEnvRecords(
|
export default async function getDecryptedEnvRecords(
|
||||||
output: Output,
|
output: Output,
|
||||||
client: Client,
|
client: Client,
|
||||||
projectId: string
|
projectId: string,
|
||||||
|
target?: ProjectEnvTarget
|
||||||
): Promise<{ envs: ProjectEnvVariable[] }> {
|
): Promise<{ envs: ProjectEnvVariable[] }> {
|
||||||
const { envs } = await getEnvRecords(output, client, projectId, {
|
const { envs } = await getEnvRecords(output, client, projectId, {
|
||||||
target: ProjectEnvTarget.Development,
|
target: target || ProjectEnvTarget.Development,
|
||||||
decrypt: true,
|
decrypt: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ export async function staticFiles(
|
|||||||
// but we don't ignore if the user is explicitly listing files
|
// but we don't ignore if the user is explicitly listing files
|
||||||
// under the now namespace, or using files in combination with gitignore
|
// under the now namespace, or using files in combination with gitignore
|
||||||
const accepts = (file: string) => {
|
const accepts = (file: string) => {
|
||||||
const relativePath = file.substr(prefixLength);
|
const relativePath = file.slice(prefixLength);
|
||||||
|
|
||||||
if (relativePath === '') {
|
if (relativePath === '') {
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -5,6 +5,6 @@ export default function humanizePath(path: string) {
|
|||||||
const resolved = resolve(path);
|
const resolved = resolve(path);
|
||||||
const _homedir = homedir();
|
const _homedir = homedir();
|
||||||
return resolved.indexOf(_homedir) === 0
|
return resolved.indexOf(_homedir) === 0
|
||||||
? `~${resolved.substr(_homedir.length)}`
|
? `~${resolved.slice(_homedir.length)}`
|
||||||
: resolved;
|
: resolved;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -160,7 +160,6 @@ export default class Now extends EventEmitter {
|
|||||||
uploadStamp,
|
uploadStamp,
|
||||||
deployStamp,
|
deployStamp,
|
||||||
quiet,
|
quiet,
|
||||||
nowConfig,
|
|
||||||
force: forceNew,
|
force: forceNew,
|
||||||
withCache,
|
withCache,
|
||||||
org,
|
org,
|
||||||
|
|||||||
@@ -19,19 +19,24 @@ export default async function inputProject(
|
|||||||
// attempt to auto-detect a project to link
|
// attempt to auto-detect a project to link
|
||||||
let detectedProject = null;
|
let detectedProject = null;
|
||||||
output.spinner('Searching for existing projects…', 1000);
|
output.spinner('Searching for existing projects…', 1000);
|
||||||
try {
|
|
||||||
const [project, slugifiedProject] = await Promise.all([
|
const [project, slugifiedProject] = await Promise.all([
|
||||||
getProjectByIdOrName(client, detectedProjectName, org.id),
|
getProjectByIdOrName(client, detectedProjectName, org.id),
|
||||||
slugifiedName !== detectedProjectName
|
slugifiedName !== detectedProjectName
|
||||||
? getProjectByIdOrName(client, slugifiedName, org.id)
|
? getProjectByIdOrName(client, slugifiedName, org.id)
|
||||||
: null,
|
: null,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
detectedProject = !(project instanceof ProjectNotFound)
|
detectedProject = !(project instanceof ProjectNotFound)
|
||||||
? project
|
? project
|
||||||
: !(slugifiedProject instanceof ProjectNotFound)
|
: !(slugifiedProject instanceof ProjectNotFound)
|
||||||
? slugifiedProject
|
? slugifiedProject
|
||||||
: null;
|
: null;
|
||||||
} catch (error) {}
|
|
||||||
|
if (detectedProject && !detectedProject.id) {
|
||||||
|
throw new Error(`Detected linked project does not have "id".`);
|
||||||
|
}
|
||||||
|
|
||||||
output.stopSpinner();
|
output.stopSpinner();
|
||||||
|
|
||||||
if (autoConfirm) {
|
if (autoConfirm) {
|
||||||
|
|||||||
@@ -180,22 +180,22 @@ export default function text({
|
|||||||
}
|
}
|
||||||
char = placeholder[value.length + caretOffset];
|
char = placeholder[value.length + caretOffset];
|
||||||
value =
|
value =
|
||||||
value.substr(0, value.length + caretOffset - 2) +
|
value.substring(0, value.length + caretOffset - 2) +
|
||||||
char +
|
char +
|
||||||
value.substr(value.length + caretOffset - 1);
|
value.slice(value.length + caretOffset - 1);
|
||||||
caretOffset--;
|
caretOffset--;
|
||||||
} else {
|
} else {
|
||||||
char = placeholder[value.length + caretOffset - 1];
|
char = placeholder[value.length + caretOffset - 1];
|
||||||
value =
|
value =
|
||||||
value.substr(0, value.length + caretOffset - 1) +
|
value.substring(0, value.length + caretOffset - 1) +
|
||||||
char +
|
char +
|
||||||
value.substr(value.length + caretOffset);
|
value.slice(value.length + caretOffset);
|
||||||
}
|
}
|
||||||
caretOffset--;
|
caretOffset--;
|
||||||
} else {
|
} else {
|
||||||
value =
|
value =
|
||||||
value.substr(0, value.length + caretOffset - 1) +
|
value.substring(0, value.length + caretOffset - 1) +
|
||||||
value.substr(value.length + caretOffset);
|
value.slice(value.length + caretOffset);
|
||||||
}
|
}
|
||||||
suggestion = '';
|
suggestion = '';
|
||||||
} else if (resolveChars.has(data)) {
|
} else if (resolveChars.has(data)) {
|
||||||
@@ -222,9 +222,9 @@ export default function text({
|
|||||||
return;
|
return;
|
||||||
} else if (!ansiRegex().test(data)) {
|
} else if (!ansiRegex().test(data)) {
|
||||||
let tmp =
|
let tmp =
|
||||||
value.substr(0, value.length + caretOffset) +
|
value.substring(0, value.length + caretOffset) +
|
||||||
data +
|
data +
|
||||||
value.substr(value.length + caretOffset);
|
value.slice(value.length + caretOffset);
|
||||||
|
|
||||||
if (mask) {
|
if (mask) {
|
||||||
if (/\d/.test(data) && caretOffset !== 0) {
|
if (/\d/.test(data) && caretOffset !== 0) {
|
||||||
@@ -236,9 +236,9 @@ export default function text({
|
|||||||
|
|
||||||
if (value[value.length + caretOffset + 1] === ' ') {
|
if (value[value.length + caretOffset + 1] === ' ') {
|
||||||
tmp =
|
tmp =
|
||||||
value.substr(0, value.length + caretOffset) +
|
value.substring(0, value.length + caretOffset) +
|
||||||
formattedData +
|
formattedData +
|
||||||
value.substr(value.length + caretOffset + formattedData.length);
|
value.slice(value.length + caretOffset + formattedData.length);
|
||||||
|
|
||||||
caretOffset += formattedData.length + 1;
|
caretOffset += formattedData.length + 1;
|
||||||
|
|
||||||
@@ -247,9 +247,9 @@ export default function text({
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
tmp =
|
tmp =
|
||||||
value.substr(0, value.length + caretOffset) +
|
value.substring(0, value.length + caretOffset) +
|
||||||
formattedData +
|
formattedData +
|
||||||
value.substr(value.length + caretOffset + formattedData.length);
|
value.slice(value.length + caretOffset + formattedData.length);
|
||||||
|
|
||||||
caretOffset += formattedData.length;
|
caretOffset += formattedData.length;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -227,6 +227,7 @@ export default async function setupAndLink(
|
|||||||
return { status: 'linked', org, project };
|
return { status: 'linked', org, project };
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
handleError(err);
|
handleError(err);
|
||||||
|
|
||||||
return { status: 'error', exitCode: 1 };
|
return { status: 'error', exitCode: 1 };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ export default async function getProjectByNameOrId(
|
|||||||
) {
|
) {
|
||||||
try {
|
try {
|
||||||
const project = await client.fetch<Project>(
|
const project = await client.fetch<Project>(
|
||||||
`/projects/${encodeURIComponent(projectNameOrId)}`,
|
`/v8/projects/${encodeURIComponent(projectNameOrId)}`,
|
||||||
{ accountId }
|
{ accountId }
|
||||||
);
|
);
|
||||||
return project;
|
return project;
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user