Compare commits

...

84 Commits

Author SHA1 Message Date
Nathan Rajlich
4c0055eaf0 Publish Stable
- vercel@21.3.2
 - @vercel/python@2.0.0
2021-03-02 11:03:53 -08:00
Nathan Rajlich
910a905192 Remove @vercel/python stable publish blacklist (#5920) 2021-03-02 11:01:02 -08:00
Steven
156a9be06e Publish Stable
- vercel@21.3.1
 - @vercel/go@1.2.1
2021-02-26 16:21:51 -05:00
Steven
e2132ee36d Publish Canary
- vercel@21.3.1-canary.0
 - @vercel/go@1.2.1-canary.0
2021-02-26 15:22:59 -05:00
Steven
4572230c1d [go] Fix analyze ignore list (#5906)
A regression was introduced in #5873 that caused the analyze step to parse Go's internal source files (eg `golang/test/bombad.go`) instead of only parsing the user's source code (eg `api/users.go`).

This resulted in the error:

```
Failed to parse AST for "api/users.go"
Error: Command failed: /vercel/1ab928d537d26157/.build-utils/.builder/node_modules/@vercel/go/dist/analyze -modpath=/vercel/workpath1 /vercel/workpath1/api/users.go
2021/02/26 14:26:42 Could not parse Go file "/vercel/workpath1/.vercel/cache/golang/test/bombad.go"
```
2021-02-26 20:15:34 +00:00
Steven
8cfac4cf86 Publish Stable
- @vercel/frameworks@0.3.0
 - @vercel/build-utils@2.10.0
 - vercel@21.3.0
 - @vercel/client@9.0.7
 - @vercel/go@1.2.0
 - @vercel/routing-utils@1.10.0
2021-02-25 19:53:07 -05:00
Steven
4f20783000 Publish Canary
- vercel@21.2.4-canary.14
2021-02-25 17:47:42 -05:00
Mark Glagola
968b7c3fb5 [cli] Use inspectorUrl from api (#5904)
Uses inspector url from api rather than generating one locally.

### 📋 Checklist

#### Tests

- [ ] The code changed/added as part of this PR has been covered with tests
- [ ] All tests pass locally with `yarn test-unit`

#### Code Review

- [ ] This PR has a concise title and thorough description useful to a reviewer
- [ ] Issue from task tracker has a link to this PR
2021-02-25 22:29:19 +00:00
Steven
407c4ec5f4 Publish Canary
- @vercel/build-utils@2.9.1-canary.10
 - vercel@21.2.4-canary.13
 - @vercel/client@9.0.7-canary.10
2021-02-24 17:52:33 -05:00
Steven
bcf393d125 [build-utils] Fix warning for canary (#5892)
### Related Issues

Follow up to #5888

### 📋 Checklist

<!--
  Please keep your PR as a Draft until the checklist is complete
-->

#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-02-24 17:52:03 -05:00
Steven
115ae0a229 Publish Canary
- @vercel/build-utils@2.9.1-canary.9
 - vercel@21.2.4-canary.12
 - @vercel/client@9.0.7-canary.9
2021-02-24 08:43:24 -05:00
Steven
d149489c9e [build-utils] Update warning for api + pages/api (#5888)
This PR updates the warning for `api` + `pages/api` to only be printed when Next.js is mixed with `@vercel/node` functions. It should not print the warning with `@vercel/go` functions for example.

### 📋 Checklist

#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-02-24 01:12:14 +00:00
Steven
5b2e6052fc Publish Canary
- vercel@21.2.4-canary.11
 - @vercel/go@1.1.9-canary.1
2021-02-23 17:45:14 -05:00
Steven
38cb5a3b99 [go] Add Go version selection via go.mod (#5873)
### 📋 Checklist

<!--
  Please keep your PR as a Draft until the checklist is complete
-->

#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-02-23 16:41:59 -05:00
Steven
84e828a0ca Publish Canary
- @vercel/frameworks@0.2.1-canary.7
 - @vercel/build-utils@2.9.1-canary.8
 - vercel@21.2.4-canary.10
 - @vercel/client@9.0.7-canary.8
2021-02-23 13:13:43 -05:00
Steven
157ce5346d [build-utils] Update Node.js 10 deprecation message (#5881)
This PR updates the discontinued date according to the changelog.

It also conditionally prints a hint to update `engines` if present or update Project Settings.

https://vercel.com/changelog/node-js-10-is-being-deprecated
2021-02-23 18:12:43 +00:00
Nathan Rajlich
0256157391 [examples] Fix "nuxtjs" example by adding a yarn.lock file (#5882)
`@babel/preset-env` shipped a change that broke Nuxt.js usage of the
module, so here we add a `yarn.lock` file that pins the preset-env
version to 7.12.17.

See: https://github.com/nuxt/nuxt.js/issues/8882
2021-02-22 18:03:50 -08:00
Steven
a45b3d0982 [examples] Update jekyll and middleman (#5871)
This PR updates the examples for jekyll and middleman to use the latest version.

I confirmed these examples work with both stable and canary.
2021-02-22 10:14:32 -05:00
Connor Davis
26af6dbc03 [tests] Update 502 to 504 for Serverless Function timeout (#5874)
We now return a 504 for lambda timeouts

### Related Issues

Related to https://github.com/vercel/now-proxy/pull/1970

#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [ ] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
2021-02-20 17:04:29 +00:00
Mark Glagola
38130103a0 [cli] Replace now.sh tests with vercel.app 2021-02-19 19:22:51 -05:00
Steven
978485818a Publish Canary
- @vercel/build-utils@2.9.1-canary.7
 - vercel@21.2.4-canary.9
 - @vercel/client@9.0.7-canary.7
2021-02-19 14:38:17 -05:00
Steven
0270784cbb [build-utils] Remove flags from bundle install (#5865)
This PR removes the flags from `bundle install` which fixes the deprecated warnings such as 

```
[DEPRECATED] The `--no-prune` flag is deprecated because it relies on being remembered across bundler invocations, which bundler will no longer do in future versions. Instead please use `bundle config set --local no_prune 'true'`, and stop using this flag
```

These flags already represent the defaults for `bundle install` anyway and we can pass environment variables in the spawn options instead.
2021-02-19 19:37:39 +00:00
Naoyuki Kanezawa
345e514924 [build-utils] Fix glob to find symlinks pointing to a directory (#5870)
https://app.clubhouse.io/vercel/story/16876

This will fix the issue that symlinks pointing to a directory is not returned on `prepareCache` and node modules linked to local files don't work.

### Related Issues

### 📋 Checklist

<!--
  Please keep your PR as a Draft until the checklist is complete
-->

#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR

Co-authored-by: Steven <steven@ceriously.com>
2021-02-19 12:43:52 -05:00
Steven
df62ec6ed0 [cli] Disable Next.js gif test and Node.js 10 tests (#5867)
* Disable gif optimization test

* Disable Node 10 tests
2021-02-19 11:20:35 -05:00
Steven
f7f81fb896 Publish Canary
- @vercel/frameworks@0.2.1-canary.6
 - @vercel/build-utils@2.9.1-canary.6
 - vercel@21.2.4-canary.8
 - @vercel/client@9.0.7-canary.6
2021-02-18 09:53:11 -05:00
Steven
a210c6e4f0 [frameworks] Add cachePattern to jekyll and middleman (#5862)
Cache the results of `bundle install`.

Related to #3908
2021-02-18 09:52:22 -05:00
Steven
50080e4b92 Publish Canary
- @vercel/frameworks@0.2.1-canary.5
 - @vercel/build-utils@2.9.1-canary.5
 - vercel@21.2.4-canary.7
 - @vercel/client@9.0.7-canary.5
 - @vercel/go@1.1.9-canary.0
2021-02-17 19:48:18 -05:00
Steven
5845bebe2d [go] Update to version 1.16 (#5858)
https://blog.golang.org/go1.16

[ch19891]
2021-02-18 00:39:09 +00:00
Steven
3e18146846 [frameworks][examples] Update RedwoodJS to 0.25.0 (#5842)
RedwoodJS [v0.25.0](https://github.com/redwoodjs/redwood/releases/tag/v0.25.0) changed to a different build command: [yarn rw deploy vercel](https://redwoodjs.com/docs/cli-commands#vercel)

This also updates the example to use the latest template from `created-redwood-app`.
2021-02-17 16:02:53 -05:00
Steven
dbde60f47f [tests] Add test for animated gif bypass (#5851)
Animated gif should bypass (serve as-is).

This is to avoid the time it takes to optimize animated images which can sometimes take minutes.

It also matches how `next dev` works so we want to stay consistent (this tests both dev and a deployment).
2021-02-16 12:16:45 -05:00
luc
aa3db23cad Publish Canary
- vercel@21.2.4-canary.6
2021-02-16 15:45:00 +01:00
Luc Leray
60a5b0a586 [cli] Use POST /projects for vc projects add (#5824)
* replace "ensure project" with "create project"

* Revert "replace "ensure project" with "create project""

This reverts commit c6955d5abda5eb9e1140d5d1d76ec7a4b193fe90.

* add test

* upsert project
2021-02-15 21:29:23 +01:00
Nathan Rajlich
8b682ccc41 Publish Canary
- @vercel/frameworks@0.2.1-canary.4
 - @vercel/build-utils@2.9.1-canary.4
 - vercel@21.2.4-canary.5
 - @vercel/client@9.0.7-canary.4
2021-02-12 18:08:29 -08:00
Nathan Rajlich
50055963af [frameworks] Fix "zola" framework detection (#5844)
`"hugo"` was a false-positive.
2021-02-12 18:07:53 -08:00
Nathan Rajlich
43fa6e6d97 Publish Canary
- @vercel/frameworks@0.2.1-canary.3
 - @vercel/build-utils@2.9.1-canary.3
 - vercel@21.2.4-canary.4
 - @vercel/client@9.0.7-canary.3
2021-02-12 16:08:49 -08:00
Nathan Rajlich
98c0b9b573 [frameworks] Add "detectors" for Zola framework (#5843) 2021-02-12 16:04:25 -08:00
Nathan Rajlich
8b8541f4aa Publish Canary
- @vercel/frameworks@0.2.1-canary.2
 - @vercel/build-utils@2.9.1-canary.2
 - vercel@21.2.4-canary.3
 - @vercel/client@9.0.7-canary.2
2021-02-12 14:22:16 -08:00
Nathan Rajlich
59cc2bf0f2 [build-utils] Inherit process.env when runPackageJsonScript() is invoking yarn (#5841)
This is a more proper fix for #5825.
2021-02-12 14:17:59 -08:00
Nathan Rajlich
30e078062f [frameworks] Add defaultVersion field (#5839)
This will be used on the docs page to render the default version that is
included in the build image.
2021-02-12 12:21:05 -08:00
Nathan Rajlich
4d015a60ef [tests] Add --network-timeout 1000000 to yarn install (#5840)
Seems like this might help with the networking issues in CI.
See https://github.com/yarnpkg/yarn/issues/4890#issuecomment-358179301
2021-02-12 12:20:48 -08:00
Nathan Rajlich
ad0e66242d [build-utils] Fix unit tests from @vercel/frameworks type change (#5829)
Fixes TypeScript error:

```
test/unit.framework-detector.test.ts:52:40 - error TS4104: The type 'readonly Framework[]' is 'readonly' and cannot be assigned to the mutable type 'Framework[]'.

52     expect(await detectFramework({ fs, frameworkList })).toBe(null);
                                          ~~~~~~~~~~~~~
```
2021-02-12 15:51:34 +00:00
Ana Jovanova
044f956de2 Publish Canary
- vercel@21.2.4-canary.2
2021-02-12 11:38:39 +01:00
Ana Trajkovska
d26ed43d26 [cli] Implement generic announcement system (#5697)
This PR improves the CLI in a way that if the API from any request returns a response which contains a header with the prefix `x-vercel-warning-*`, `x-vercel-notice-*` or `x-vercel-tip-*`, it will print out the message to the output to let the user know.

This new feature also supports additional headers which improve the message, so that a link can be printed out for more information. Those are `x-vercel-link-*`, where we specify a link, and `x-vercel-action`, where we specify the text before the link.

Here's an example for a warning message which warns the user if they add a DNS record for a domain which does not use Vercel Nameservers:

<img width="719" alt="Screenshot 2021-01-18 at 17 49 14" src="https://user-images.githubusercontent.com/35507539/104943788-8da12400-59b6-11eb-8e54-77a3c517e56d.png">

The API request from above which creates a DNS record has the following HTTP response headers set:

```
x-vercel-warning-create-domain-record: Your Domain isn't using Vercel nameservers. In order for the changes to your DNS Records to get applied, however, it needs to.
x-vercel-link-create-domain-record: https://vercel.link/configure-vercel-nameservers
```

If `x-vercel-action` is not specified, it defaults to `Learn More`.

### 📋 Checklist

<!--
  Please keep your PR as a Draft until the checklist is complete
-->

#### Tests

- [ ] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-02-12 09:28:47 +00:00
Nathan Rajlich
84d859a016 Publish Canary
- @vercel/frameworks@0.2.1-canary.1
 - @vercel/build-utils@2.9.1-canary.1
 - vercel@21.2.4-canary.1
 - @vercel/client@9.0.7-canary.1
 - @vercel/routing-utils@1.9.3-canary.0
2021-02-11 14:42:07 -08:00
Nathan Rajlich
9238687089 [frameworks] Restore correct order of frameworks for detection (#5828) 2021-02-11 14:41:20 -08:00
Nathan Rajlich
212c91cf32 [frameworks] Add type annotations (#5819)
* [frameworks] Add type annotations

This will be necessary for static type analysis in our API docs generator.

* Apply suggestions from code review

Co-authored-by: Steven <steven@ceriously.com>

* Add quotes to @examples

Co-authored-by: Steven <steven@ceriously.com>
2021-02-11 11:42:13 -08:00
Nathan Rajlich
48b34f5b02 [frameworks] Add test case to ensure all "demo" URLs are public Vercel deployments (#5818)
All demo URLs must be public for the template import flow, so this test case ensures that we don't accidentally publish a demo URL that is not public.
2021-02-10 22:35:08 +00:00
Nathan Rajlich
b2c72538a3 [frameworks] Export frameworks array with as const (#5817)
This will be necessary for static type analysis in our API docs generator.

The default export is still typed as `Framework[]` for backwards-compat purposes, but the `frameworks` export is the statically typed array.
2021-02-10 21:40:05 +00:00
Nathan Rajlich
57d8702188 [build-utils][cli] Fix frameworks usage (#5811)
Fixes unit test failures like: https://github.com/vercel/vercel/runs/1867209532#step:11:191 from PR #5740
2021-02-10 04:14:02 +00:00
Allen Hai
9a72195d8a [routing-utils] add user boolean prop to route schema (#5788)
This PR adds a boolean `user` property to help determine which routes where added by the user and which routes were added by the platform.
2021-02-10 01:42:50 +00:00
Nathan Rajlich
29ab7f7cfb Publish Canary
- @vercel/frameworks@0.2.1-canary.0
 - @vercel/build-utils@2.9.1-canary.0
 - vercel@21.2.4-canary.0
 - @vercel/client@9.0.7-canary.0
2021-02-09 15:13:16 -08:00
Nathan Rajlich
ffa36c12d5 [frameworks] Convert to TypeScript (#5740)
This PR converts the `frameworks.json` file to TypeScript, and extends the values with the detection logic from `@vercel/static-build`, so that it's publicly editable. You also don't need to do the type casting downstream anymore.

As a consequence, it also makes Zola a 1st-class framework, as it was previously missing from the `frameworks.json` file, but present in the static-build frameworks. An example has been included based on their "Getting Started" tutorial.

CH-3808
CH-18771
2021-02-09 22:55:49 +00:00
Nathan Rajlich
d1fc729d31 [cli] Use sleep() util function in vc dev (#5784) 2021-02-08 17:20:47 -08:00
Nathan Rajlich
ba47ad00f9 [examples][tests] Update "gridsome" and "hexo" for Node 14 (#5802)
The version of `sharp` used in gridsome 0.6.0 does not compile on Node 14.
2021-02-08 19:02:39 -05:00
Steven
2b62de803f Publish Stable
- @vercel/build-utils@2.9.0
 - vercel@21.2.3
 - @vercel/client@9.0.6
2021-02-04 15:43:26 -05:00
Steven
196433ec9d Publish Canary
- @vercel/build-utils@2.8.1-canary.1
 - vercel@21.2.3-canary.0
 - @vercel/client@9.0.6-canary.4
2021-02-04 14:26:59 -05:00
Steven
0cdd571087 [build-utils] Add support for Node.js 14, warn for Node.js 10 (#5761)
### Related

Related to #4754 

### Upstream

https://aws.amazon.com/blogs/compute/node-js-14-x-runtime-now-available-in-aws-lambda/

### 📋 Checklist


#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-02-04 14:25:16 -05:00
Steven
5e2227507d [docs] Fix links to GH Discussion categories (#5777) 2021-02-01 12:08:57 -05:00
Nathan Rajlich
e3d478a471 Publish Stable
- vercel@21.2.2
 - @vercel/go@1.1.8
2021-01-29 16:57:35 -08:00
Steven
187d41434a Publish Canary
- vercel@21.2.2-canary.1
2021-01-29 18:55:54 -05:00
Steven
f6fde2d351 [cli] Update error message when project is not found (#5770)
Both `vc` and `vc dev` already ask to link/setup when the project is not found, so this will update the error message to mention the possible reasons why a project was not found.

### Before

```
$ vc
Vercel CLI 21.2.1
️ Your project was either removed from Vercel or you’re not a member of it anymore.
? Set up and deploy “~/Code/myproject”? [Y/n]
```

### After

```
$ vc
Vercel CLI 21.2.1
️ Your Project was either deleted, transferred to a new Team, or you don’t have access to it anymore.
? Set up and deploy “~/Code/myproject”? [Y/n]
```


### 📋 Checklist



#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-01-29 18:54:30 -05:00
Steven
e09ceca95d Publish Canary
- @vercel/build-utils@2.8.1-canary.0
 - @vercel/cgi@1.0.7-canary.0
 - vercel@21.2.2-canary.0
 - @vercel/client@9.0.6-canary.3
 - @vercel/go@1.1.8-canary.0
 - @vercel/node@1.9.1-canary.0
 - @vercel/python@1.2.5-canary.1
 - @vercel/ruby@1.2.6-canary.0
2021-01-29 12:40:38 -05:00
Igor Klopov
d9318af654 [go] support skipDownload flag (#5767) 2021-01-29 12:37:06 -05:00
Nathan Rajlich
679cd6f14e [cli] Use output.stopSpinner() interface consistently (#5762)
There was an issue with `output.spinner()` spinner if the old-style
interface was used (invoking the return value to stop the spinner),
which caused the internal spinner instance in `output` to be defined,
but in a "stopped" state. Then later the `text` gets updated to a
different value but the spinner is still stopped and nothing gets
rendered.

The fix is to ensure that `output.stopSpinner()` is called rather
than directly invoking the cancel function. In fact, the return
value of `output.spinner()` has been removed completely.
2021-01-28 14:16:17 -08:00
Nathan Rajlich
47c7a13324 [all] Rewrite build scripts in JavaScript (#5763)
Fixes these strange Windows failures that randomly started happening:

https://github.com/vercel/vercel/pull/5761/checks?check_run_id=1780742477
2021-01-28 19:20:07 +00:00
Nathan Rajlich
f498d25d8b Publish Stable
- vercel@21.2.1
2021-01-27 14:26:52 -08:00
Nathan Rajlich
726d3924ae Publish Canary
- vercel@21.2.1-canary.0
 - @vercel/python@1.2.5-canary.0
2021-01-26 18:31:12 -08:00
Nathan Rajlich
d3e6c2d335 [python] Fix request URL path encoding for basic, WSGI and ASGI interfaces (#5739)
`@vercel/python` has been incorrectly "unquoting" the incoming request
URL for all Python interfaces (basic, WSGI, ASGI). The proper behavior is
to pass along the request URL to the serverless function handler unmodified.

This is proven by the additional test that is added in this PR, which actually
fires up the corresponding Python server via i.e. `python3 api/asgi.py`,
and executes the test probes defined in the
`test/fixtures/00-request-path/vercel.json` file. Adding this test ensures
that the request URL behavior of `@vercel/python` matches the behavior
as compared to executing via `python` directly.
2021-01-26 12:37:15 -08:00
Nathan Rajlich
f9ed84a5c9 [cli] Pass the output object to vercel switch command (#5756)
Fixes error: `Cannot read property 'spinner' of undefined`
2021-01-26 12:34:08 -08:00
Nathan Rajlich
88a815b5f0 Publish Stable
- vercel@21.2.0
2021-01-25 14:44:41 -08:00
Steven
9f49743ea4 Publish Stable
- @vercel/build-utils@2.8.0
2021-01-22 18:53:38 -05:00
Steven
0ac3ae19c0 Publish Canary
- @vercel/build-utils@2.7.1-canary.2
 - vercel@21.1.1-canary.3
 - @vercel/client@9.0.6-canary.2
2021-01-22 18:48:09 -05:00
Steven
e7920fd783 [build-utils] Fix warning message (#5741) 2021-01-22 18:47:33 -05:00
Steven
b4c13470df Publish Canary
- @vercel/build-utils@2.7.1-canary.1
 - vercel@21.1.1-canary.2
 - @vercel/client@9.0.6-canary.1
2021-01-22 11:59:09 -05:00
Steven
2a797b77b9 [build-utils] Add support for config.nodeVersion (#5729)
### 📋 Checklist

<!--
  Please keep your PR as a Draft until the checklist is complete
-->

#### Tests

- [x] The code changed/added as part of this PR has been covered with tests
- [x] All tests pass locally with `yarn test-unit`

#### Code Review

- [x] This PR has a concise title and thorough description useful to a reviewer
- [x] Issue from task tracker has a link to this PR
2021-01-22 11:57:55 -05:00
Nathan Rajlich
85a34126df Publish Canary
- vercel@21.1.1-canary.1
2021-01-21 22:48:10 -08:00
Nathan Rajlich
92889c5376 [cli] Re-use a single Output instance throughout the codebase (#5722)
Previously every command implementation was invoking `createOutput()` individually, causing multiple instances of `Output` to be created, which was unnecessary, and causes issues with the shared `spinner` instance introduced by #5717.

Now, the `Output` instance created in `src/index.js` is passed down to the sub-command being executed and therefore re-used.
2021-01-21 19:57:08 -08:00
Nathan Rajlich
66a7fa30b8 [cli] Show login prompt for vercel dev (#5718)
`vercel dev` requires the user to be logged in ever since the project needs to be queried from the database.

**Before:**

<img width="682" alt="Screen Shot 2021-01-18 at 7 58 56 PM" src="https://user-images.githubusercontent.com/71256/104986545-14aac800-59c8-11eb-8302-836a5481ce5f.png">

**After:**

<img width="497" alt="Screen Shot 2021-01-18 at 7 59 01 PM" src="https://user-images.githubusercontent.com/71256/104986547-15dbf500-59c8-11eb-85f6-d0ad628c8e8d.png">
2021-01-21 19:56:35 -08:00
Nathan Rajlich
a7ba405503 [cli] Make Output class a singleton and have reusable Spinner (#5717)
This is a refactor of the `ora` Spinner usage to make the preferred usage be via the `output.spinner()` and `output.stopSpinner()` functions.

* The `Output` instance has a local spinner instance that can be updated via multiple calls to `output.spinner()`.
* `output.print()` and friends call `output.stopSpinner()` implicitly, so `output.stopSpinner()` doesn't need to be called unless it is desired to immediately remove the spinner before rendering the next text.
* Because there's meant to be a shared `spinner` for the `output` instance, it means that the same `output` instance needs to be used throughout the CLI lifecycle. Therefore, the `createOutput()` function now returns a singleton so that it's guaranteed to be the same instance.

https://user-images.githubusercontent.com/71256/105009979-0670a200-59f0-11eb-9853-232f92eae536.mov

https://user-images.githubusercontent.com/71256/105010034-17211800-59f0-11eb-918d-b15f457e9640.mov

In the "before" video above, note how there's a brief second where the spinner is cleared, but the deployment URL has not yet been printed. In the "after" video, this is fixed and the spinner is only cleared once the CLI is ready to render the deployment URL.
2021-01-19 11:38:29 -08:00
luc
84145245ba Publish Canary
- @vercel/build-utils@2.7.1-canary.0
 - vercel@21.1.1-canary.0
 - @vercel/client@9.0.6-canary.0
2021-01-19 09:29:20 +01:00
Luc Leray
90d2e8b63b [cli] Hide "Development" for secret env variables (#5710) 2021-01-19 09:27:42 +01:00
Nathan Rajlich
0d31fe8018 [cli] Fix cleanUrls: true when using outputDirectory (#5701)
This is essentially a refactor of the `@vercel/static` builder that
simplifies the code a lot and fixes a bug where `cleanUrls: true`
was not working correctly when using an `outputDirectory`.
2021-01-18 12:03:34 -08:00
Mark Glagola
37d747c241 [cli] Add auto renew prompt to domains buy (#5663)
Add auto renew prompt to `domains buy`

```diff
 > The domain "example.com" is available to buy under <username>!
 > Buy now for $20 (1yr)? [y|N] 
+ > Auto renew yearly for $20? [Y|n] 
```

### 📋 Checklist

#### Tests

- [ ] The code changed/added as part of this PR has been covered with tests
- [ ] All tests pass locally with `yarn test-unit`

#### Code Review

- [ ] This PR has a concise title and thorough description useful to a reviewer
- [ ] Issue from task tracker has a link to this PR
2021-01-15 19:13:37 +00:00
Nathan Rajlich
3f052d905f [build-utils] Remove now-next and now-static-build tests (#5695)
These directories are no longer in this repository, so the tests were failing.
2021-01-14 13:50:19 +00:00
230 changed files with 18057 additions and 7900 deletions

View File

@@ -4,8 +4,8 @@ contact_links:
url: https://vercel.com/support/request url: https://vercel.com/support/request
about: Report a bug using the Vercel support form about: Report a bug using the Vercel support form
- name: Feature Request - name: Feature Request
url: https://github.com/vercel/vercel/discussions/new?category_id=66161 url: https://github.com/vercel/vercel/discussions/new?category=ideas
about: Share ideas for new features about: Share ideas for new features
- name: Ask a Question - name: Ask a Question
url: https://github.com/vercel/vercel/discussions/new?category_id=66160 url: https://github.com/vercel/vercel/discussions/new?category=help
about: Ask the community for help about: Ask the community for help

View File

@@ -17,7 +17,7 @@ jobs:
with: with:
node-version: 12 node-version: 12
- name: Install - name: Install
run: yarn install --check-files --frozen-lockfile run: yarn install --check-files --frozen-lockfile --network-timeout 1000000
- name: Build - name: Build
run: yarn build run: yarn build
env: env:

View File

@@ -16,7 +16,7 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
os: [ubuntu-latest] os: [ubuntu-latest]
node: [10, 12] node: [12]
runs-on: ${{ matrix.os }} runs-on: ${{ matrix.os }}
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@@ -26,7 +26,7 @@ jobs:
- run: git fetch origin master --depth=100 - run: git fetch origin master --depth=100
- run: git fetch origin ${{ github.ref }} --depth=100 - run: git fetch origin ${{ github.ref }} --depth=100
- run: git diff origin/master...HEAD --name-only - run: git diff origin/master...HEAD --name-only
- run: yarn install - run: yarn install --network-timeout 1000000
- run: yarn run build - run: yarn run build
- uses: actions/setup-node@v1 - uses: actions/setup-node@v1
with: with:

View File

@@ -11,12 +11,12 @@ on:
jobs: jobs:
test: test:
name: Dev name: Dev
timeout-minutes: 45 timeout-minutes: 60
strategy: strategy:
fail-fast: false fail-fast: false
matrix: matrix:
os: [ubuntu-latest, macos-latest] os: [ubuntu-latest, macos-latest]
node: [10, 12] node: [12]
runs-on: ${{ matrix.os }} runs-on: ${{ matrix.os }}
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@@ -29,7 +29,7 @@ jobs:
- name: Install Hugo - name: Install Hugo
if: matrix.os == 'macos-latest' if: matrix.os == 'macos-latest'
run: curl -L -O https://github.com/gohugoio/hugo/releases/download/v0.56.0/hugo_0.56.0_macOS-64bit.tar.gz && tar -xzf hugo_0.56.0_macOS-64bit.tar.gz && mv ./hugo packages/now-cli/test/dev/fixtures/08-hugo/ run: curl -L -O https://github.com/gohugoio/hugo/releases/download/v0.56.0/hugo_0.56.0_macOS-64bit.tar.gz && tar -xzf hugo_0.56.0_macOS-64bit.tar.gz && mv ./hugo packages/now-cli/test/dev/fixtures/08-hugo/
- run: yarn install - run: yarn install --network-timeout 1000000
- run: yarn run build - run: yarn run build
- uses: actions/setup-node@v1 - uses: actions/setup-node@v1
with: with:

View File

@@ -21,7 +21,7 @@ jobs:
- run: git fetch origin master --depth=100 - run: git fetch origin master --depth=100
- run: git fetch origin ${{ github.ref }} --depth=100 - run: git fetch origin ${{ github.ref }} --depth=100
- run: git diff origin/master...HEAD --name-only - run: git diff origin/master...HEAD --name-only
- run: yarn install - run: yarn install --network-timeout 1000000
- run: yarn run build - run: yarn run build
- run: yarn test-integration-once --clean false - run: yarn test-integration-once --clean false
env: env:

View File

@@ -16,7 +16,7 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
os: [ubuntu-latest, macos-latest, windows-latest] os: [ubuntu-latest, macos-latest, windows-latest]
node: [10, 12] node: [12]
runs-on: ${{ matrix.os }} runs-on: ${{ matrix.os }}
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@@ -29,7 +29,7 @@ jobs:
- uses: actions/setup-node@v1 - uses: actions/setup-node@v1
with: with:
node-version: ${{ matrix.node }} node-version: ${{ matrix.node }}
- run: yarn install - run: yarn install --network-timeout 1000000
- run: yarn run build - run: yarn run build
- run: yarn run lint - run: yarn run lint
if: matrix.os == 'ubuntu-latest' && matrix.node == 12 # only run lint once if: matrix.os == 'ubuntu-latest' && matrix.node == 12 # only run lint once

View File

@@ -12,6 +12,11 @@ const frameworks = (_frameworks as Framework[])
...frameworkItem, ...frameworkItem,
detectors: undefined, detectors: undefined,
sort: undefined, sort: undefined,
dependency: undefined,
defaultRoutes: undefined,
cachePattern: undefined,
devCommand: undefined,
buildCommand: undefined,
}; };
if (framework.logo) { if (framework.logo) {

View File

@@ -5,7 +5,7 @@
"description": "API for the vercel/vercel repo", "description": "API for the vercel/vercel repo",
"main": "index.js", "main": "index.js",
"scripts": { "scripts": {
"build": "yarn --cwd .. && node ../utils/run.js build all" "vercel-build": "yarn --cwd .. && node ../utils/run.js build all"
}, },
"dependencies": { "dependencies": {
"@sentry/node": "5.11.1", "@sentry/node": "5.11.1",

View File

@@ -5,6 +5,7 @@
"strict": false, "strict": false,
"forceConsistentCasingInFileNames": true, "forceConsistentCasingInFileNames": true,
"noEmit": true, "noEmit": true,
"noEmitOnError": true,
"esModuleInterop": true, "esModuleInterop": true,
"module": "commonjs", "module": "commonjs",
"moduleResolution": "node", "moduleResolution": "node",

View File

@@ -8,6 +8,6 @@
"explore": "gridsome explore" "explore": "gridsome explore"
}, },
"dependencies": { "dependencies": {
"gridsome": "^0.6.0" "gridsome": "0.7.23"
} }
} }

View File

@@ -3,18 +3,18 @@
"version": "0.0.0", "version": "0.0.0",
"private": true, "private": true,
"hexo": { "hexo": {
"version": "3.9.0" "version": "5.3.0"
}, },
"dependencies": { "dependencies": {
"hexo": "^3.9.0", "hexo": "^5.3.0",
"hexo-generator-archive": "^0.1.5", "hexo-generator-archive": "^1.0.0",
"hexo-generator-category": "^0.1.3", "hexo-generator-category": "^1.0.0",
"hexo-generator-index": "^0.2.1", "hexo-generator-index": "^2.0.0",
"hexo-generator-tag": "^0.2.0", "hexo-generator-tag": "^1.0.0",
"hexo-renderer-ejs": "^0.3.1", "hexo-renderer-ejs": "^1.0.0",
"hexo-renderer-stylus": "^0.3.3", "hexo-renderer-marked": "^3.3.0",
"hexo-renderer-marked": "^1.0.1", "hexo-renderer-stylus": "^2.0.1",
"hexo-server": "^0.3.3" "hexo-server": "^2.0.0"
}, },
"scripts": { "scripts": {
"dev": "hexo server -p $PORT", "dev": "hexo server -p $PORT",

View File

@@ -1,6 +1,9 @@
_site _site
public
.sass-cache .sass-cache
.jekyll-cache
.jekyll-metadata .jekyll-metadata
vendor
public
.env .env
.env.build .env.build
.vercel

View File

@@ -1,2 +0,0 @@
README.md
yarn.lock

View File

@@ -1,4 +1,5 @@
--- ---
permalink: /404.html
layout: default layout: default
--- ---

View File

@@ -1,5 +1,4 @@
source "https://rubygems.org" source "https://rubygems.org"
# Hello! This is where you manage which Jekyll version is used to run. # Hello! This is where you manage which Jekyll version is used to run.
# When you want to use a different version, change it below, save the # When you want to use a different version, change it below, save the
# file and run `bundle install`. Run Jekyll with `bundle exec`, like so: # file and run `bundle install`. Run Jekyll with `bundle exec`, like so:
@@ -8,27 +7,23 @@ source "https://rubygems.org"
# #
# This will help ensure the proper Jekyll version is running. # This will help ensure the proper Jekyll version is running.
# Happy Jekylling! # Happy Jekylling!
gem "jekyll", "~> 3.8.6" gem "jekyll", "~> 4.2.0"
# This is the default theme for new Jekyll sites. You may change this to anything you like. # This is the default theme for new Jekyll sites. You may change this to anything you like.
gem "minima", "~> 2.0" gem "minima", "~> 2.5"
# If you want to use GitHub Pages, remove the "gem "jekyll"" above and # If you want to use GitHub Pages, remove the "gem "jekyll"" above and
# uncomment the line below. To upgrade, run `bundle update github-pages`. # uncomment the line below. To upgrade, run `bundle update github-pages`.
# gem "github-pages", group: :jekyll_plugins # gem "github-pages", group: :jekyll_plugins
# If you have any plugins, put them here! # If you have any plugins, put them here!
group :jekyll_plugins do group :jekyll_plugins do
gem "jekyll-feed", "~> 0.6" gem "jekyll-feed", "~> 0.12"
end end
# Windows does not include zoneinfo files, so bundle the tzinfo-data gem # Windows and JRuby does not include zoneinfo files, so bundle the tzinfo-data gem
# and associated library. # and associated library.
install_if -> { RUBY_PLATFORM =~ %r!mingw|mswin|java! } do platforms :mingw, :x64_mingw, :mswin, :jruby do
gem "tzinfo", "~> 1.2" gem "tzinfo", "~> 1.2"
gem "tzinfo-data" gem "tzinfo-data"
end end
# Performance-booster for watching directories on Windows # Performance-booster for watching directories on Windows
gem "wdm", "~> 0.1.0", :install_if => Gem.win_platform? gem "wdm", "~> 0.1.1", :platforms => [:mingw, :x64_mingw, :mswin]

View File

@@ -1,82 +1,80 @@
GEM GEM
remote: https://rubygems.org/ remote: https://rubygems.org/
specs: specs:
addressable (2.6.0) addressable (2.7.0)
public_suffix (>= 2.0.2, < 4.0) public_suffix (>= 2.0.2, < 5.0)
colorator (1.1.0) colorator (1.1.0)
concurrent-ruby (1.1.5) concurrent-ruby (1.1.8)
em-websocket (0.5.1) em-websocket (0.5.2)
eventmachine (>= 0.12.9) eventmachine (>= 0.12.9)
http_parser.rb (~> 0.6.0) http_parser.rb (~> 0.6.0)
eventmachine (1.2.7) eventmachine (1.2.7)
ffi (1.11.1) ffi (1.14.2)
forwardable-extended (2.6.0) forwardable-extended (2.6.0)
http_parser.rb (0.6.0) http_parser.rb (0.6.0)
i18n (0.9.5) i18n (1.8.9)
concurrent-ruby (~> 1.0) concurrent-ruby (~> 1.0)
jekyll (3.8.6) jekyll (4.2.0)
addressable (~> 2.4) addressable (~> 2.4)
colorator (~> 1.0) colorator (~> 1.0)
em-websocket (~> 0.5) em-websocket (~> 0.5)
i18n (~> 0.7) i18n (~> 1.0)
jekyll-sass-converter (~> 1.0) jekyll-sass-converter (~> 2.0)
jekyll-watch (~> 2.0) jekyll-watch (~> 2.0)
kramdown (~> 1.14) kramdown (~> 2.3)
kramdown-parser-gfm (~> 1.0)
liquid (~> 4.0) liquid (~> 4.0)
mercenary (~> 0.3.3) mercenary (~> 0.4.0)
pathutil (~> 0.9) pathutil (~> 0.9)
rouge (>= 1.7, < 4) rouge (~> 3.0)
safe_yaml (~> 1.0) safe_yaml (~> 1.0)
jekyll-feed (0.12.1) terminal-table (~> 2.0)
jekyll-feed (0.15.1)
jekyll (>= 3.7, < 5.0) jekyll (>= 3.7, < 5.0)
jekyll-sass-converter (1.5.2) jekyll-sass-converter (2.1.0)
sass (~> 3.4) sassc (> 2.0.1, < 3.0)
jekyll-seo-tag (2.6.1) jekyll-seo-tag (2.7.1)
jekyll (>= 3.3, < 5.0) jekyll (>= 3.8, < 5.0)
jekyll-watch (2.2.1) jekyll-watch (2.2.1)
listen (~> 3.0) listen (~> 3.0)
kramdown (1.17.0) kramdown (2.3.0)
rexml
kramdown-parser-gfm (1.1.0)
kramdown (~> 2.0)
liquid (4.0.3) liquid (4.0.3)
listen (3.1.5) listen (3.4.1)
rb-fsevent (~> 0.9, >= 0.9.4) rb-fsevent (~> 0.10, >= 0.10.3)
rb-inotify (~> 0.9, >= 0.9.7) rb-inotify (~> 0.9, >= 0.9.10)
ruby_dep (~> 1.2) mercenary (0.4.0)
mercenary (0.3.6) minima (2.5.1)
minima (2.5.0) jekyll (>= 3.5, < 5.0)
jekyll (~> 3.5)
jekyll-feed (~> 0.9) jekyll-feed (~> 0.9)
jekyll-seo-tag (~> 2.1) jekyll-seo-tag (~> 2.1)
pathutil (0.16.2) pathutil (0.16.2)
forwardable-extended (~> 2.6) forwardable-extended (~> 2.6)
public_suffix (3.1.1) public_suffix (4.0.6)
rb-fsevent (0.10.3) rb-fsevent (0.10.4)
rb-inotify (0.10.0) rb-inotify (0.10.1)
ffi (~> 1.0) ffi (~> 1.0)
rouge (3.6.0) rexml (3.2.4)
ruby_dep (1.5.0) rouge (3.26.0)
safe_yaml (1.0.5) safe_yaml (1.0.5)
sass (3.7.4) sassc (2.4.0)
sass-listen (~> 4.0.0) ffi (~> 1.9)
sass-listen (4.0.0) terminal-table (2.0.0)
rb-fsevent (~> 0.9, >= 0.9.4) unicode-display_width (~> 1.1, >= 1.1.1)
rb-inotify (~> 0.9, >= 0.9.7) unicode-display_width (1.7.0)
thread_safe (0.3.6)
tzinfo (1.2.5)
thread_safe (~> 0.1)
tzinfo-data (1.2019.2)
tzinfo (>= 1.0.0)
wdm (0.1.1)
PLATFORMS PLATFORMS
ruby x86_64-linux
DEPENDENCIES DEPENDENCIES
jekyll (~> 3.8.6) jekyll (~> 4.2.0)
jekyll-feed (~> 0.6) jekyll-feed (~> 0.12)
minima (~> 2.0) minima (~> 2.5)
tzinfo (~> 1.2) tzinfo (~> 1.2)
tzinfo-data tzinfo-data
wdm (~> 0.1.0) wdm (~> 0.1.1)
BUNDLED WITH BUNDLED WITH
2.1.4 2.2.4

View File

@@ -7,12 +7,17 @@
# #
# For technical reasons, this file is *NOT* reloaded automatically when you use # For technical reasons, this file is *NOT* reloaded automatically when you use
# 'bundle exec jekyll serve'. If you change this file, please restart the server process. # 'bundle exec jekyll serve'. If you change this file, please restart the server process.
#
# If you need help with YAML syntax, here are some quick references for you:
# https://learn-the-web.algonquindesign.ca/topics/markdown-yaml-cheat-sheet/#yaml
# https://learnxinyminutes.com/docs/yaml/
#
# Site settings # Site settings
# These are used to personalize your new site. If you look in the HTML files, # These are used to personalize your new site. If you look in the HTML files,
# you will see them accessed via {{ site.title }}, {{ site.email }}, and so on. # you will see them accessed via {{ site.title }}, {{ site.email }}, and so on.
# You can create any custom variable you would like, and they will be accessible # You can create any custom variable you would like, and they will be accessible
# in the templates via {{ site.myvariable }}. # in the templates via {{ site.myvariable }}.
title: Your awesome title title: Your awesome title
email: your-email@example.com email: your-email@example.com
description: >- # this means to ignore newlines until "baseurl:" description: >- # this means to ignore newlines until "baseurl:"
@@ -23,21 +28,27 @@ baseurl: "" # the subpath of your site, e.g. /blog
url: "" # the base hostname & protocol for your site, e.g. http://example.com url: "" # the base hostname & protocol for your site, e.g. http://example.com
twitter_username: jekyllrb twitter_username: jekyllrb
github_username: jekyll github_username: jekyll
permalink: pretty
# Build settings # Build settings
markdown: kramdown
theme: minima theme: minima
plugins: plugins:
- jekyll-feed - jekyll-feed
# Exclude from processing. # Exclude from processing.
# The following items will not be processed, by default. Create a custom list # The following items will not be processed, by default.
# to override the default setting. # Any item listed under the `exclude:` key here will be automatically added to
# the internal "default list".
#
# Excluded items can be processed by explicitly listing the directories or
# their entries' file path in the `include:` list.
#
# exclude: # exclude:
# - .sass-cache/
# - .jekyll-cache/
# - gemfiles/
# - Gemfile # - Gemfile
# - Gemfile.lock # - Gemfile.lock
# - node_modules # - node_modules/
# - vendor/bundle/ # - vendor/bundle/
# - vendor/cache/ # - vendor/cache/
# - vendor/gems/ # - vendor/gems/

View File

@@ -1,18 +1,23 @@
--- ---
layout: post layout: post
title: "Welcome to Jekyll!" title: 'Welcome to Jekyll!'
date: 2019-07-18 00:15:52 +0100 date: 2021-02-19 23:17:16 +0000
categories: jekyll update categories: jekyll update
--- ---
Youll find this post in your `_posts` directory. Go ahead and edit it and re-build the site to see your changes. You can rebuild the site in many different ways, but the most common way is to run `jekyll serve`, which launches a web server and auto-regenerates your site when a file is updated. Youll find this post in your `_posts` directory. Go ahead and edit it and re-build the site to see your changes. You can rebuild the site in many different ways, but the most common way is to run `jekyll serve`, which launches a web server and auto-regenerates your site when a file is updated.
To add new posts, simply add a file in the `_posts` directory that follows the convention `YYYY-MM-DD-name-of-post.ext` and includes the necessary front matter. Take a look at the source for this post to get an idea about how it works. Jekyll requires blog post files to be named according to the following format:
`YEAR-MONTH-DAY-title.MARKUP`
Where `YEAR` is a four-digit number, `MONTH` and `DAY` are both two-digit numbers, and `MARKUP` is the file extension representing the format used in the file. After that, include the necessary front matter. Take a look at the source for this post to get an idea about how it works.
Jekyll also offers powerful support for code snippets: Jekyll also offers powerful support for code snippets:
{% highlight ruby %} {% highlight ruby %}
def print_hi(name) def print_hi(name)
puts "Hi, #{name}" puts "Hi, #{name}"
end end
print_hi('Tom') print_hi('Tom')
#=> prints 'Hi, Tom' to STDOUT. #=> prints 'Hi, Tom' to STDOUT.
@@ -21,5 +26,5 @@ print_hi('Tom')
Check out the [Jekyll docs][jekyll-docs] for more info on how to get the most out of Jekyll. File all bugs/feature requests at [Jekylls GitHub repo][jekyll-gh]. If you have questions, you can ask them on [Jekyll Talk][jekyll-talk]. Check out the [Jekyll docs][jekyll-docs] for more info on how to get the most out of Jekyll. File all bugs/feature requests at [Jekylls GitHub repo][jekyll-gh]. If you have questions, you can ask them on [Jekyll Talk][jekyll-talk].
[jekyll-docs]: https://jekyllrb.com/docs/home [jekyll-docs]: https://jekyllrb.com/docs/home
[jekyll-gh]: https://github.com/jekyll/jekyll [jekyll-gh]: https://github.com/jekyll/jekyll
[jekyll-talk]: https://talk.jekyllrb.com/ [jekyll-talk]: https://talk.jekyllrb.com/

View File

@@ -3,3 +3,4 @@
.DS_Store .DS_Store
.sass-cache .sass-cache
build/ build/
.vercel

View File

@@ -1,29 +1,29 @@
GEM GEM
remote: https://rubygems.org/ remote: https://rubygems.org/
specs: specs:
activesupport (5.0.7.2) activesupport (5.2.4.5)
concurrent-ruby (~> 1.0, >= 1.0.2) concurrent-ruby (~> 1.0, >= 1.0.2)
i18n (>= 0.7, < 2) i18n (>= 0.7, < 2)
minitest (~> 5.1) minitest (~> 5.1)
tzinfo (~> 1.1) tzinfo (~> 1.1)
addressable (2.7.0) addressable (2.7.0)
public_suffix (>= 2.0.2, < 5.0) public_suffix (>= 2.0.2, < 5.0)
autoprefixer-rails (9.6.1.1) autoprefixer-rails (9.8.6.5)
execjs execjs
backports (3.15.0) backports (3.20.2)
coffee-script (2.4.1) coffee-script (2.4.1)
coffee-script-source coffee-script-source
execjs execjs
coffee-script-source (1.12.2) coffee-script-source (1.12.2)
concurrent-ruby (1.1.5) concurrent-ruby (1.1.8)
contracts (0.13.0) contracts (0.13.0)
dotenv (2.7.5) dotenv (2.7.6)
erubis (2.7.0) erubis (2.7.0)
execjs (2.7.0) execjs (2.7.0)
fast_blank (1.0.0) fast_blank (1.0.0)
fastimage (2.1.7) fastimage (2.2.2)
ffi (1.11.1) ffi (1.14.2)
haml (5.1.2) haml (5.2.1)
temple (>= 0.8.0) temple (>= 0.8.0)
tilt tilt
hamster (3.0.0) hamster (3.0.0)
@@ -31,24 +31,25 @@ GEM
hashie (3.6.0) hashie (3.6.0)
i18n (0.9.5) i18n (0.9.5)
concurrent-ruby (~> 1.0) concurrent-ruby (~> 1.0)
kramdown (1.17.0) kramdown (2.3.0)
rexml
listen (3.0.8) listen (3.0.8)
rb-fsevent (~> 0.9, >= 0.9.4) rb-fsevent (~> 0.9, >= 0.9.4)
rb-inotify (~> 0.9, >= 0.9.7) rb-inotify (~> 0.9, >= 0.9.7)
memoist (0.16.0) memoist (0.16.2)
middleman (4.3.5) middleman (4.3.11)
coffee-script (~> 2.2) coffee-script (~> 2.2)
haml (>= 4.0.5) haml (>= 4.0.5)
kramdown (~> 1.2) kramdown (>= 2.3.0)
middleman-cli (= 4.3.5) middleman-cli (= 4.3.11)
middleman-core (= 4.3.5) middleman-core (= 4.3.11)
middleman-autoprefixer (2.10.1) middleman-autoprefixer (2.10.1)
autoprefixer-rails (~> 9.1) autoprefixer-rails (~> 9.1)
middleman-core (>= 3.3.3) middleman-core (>= 3.3.3)
middleman-cli (4.3.5) middleman-cli (4.3.11)
thor (>= 0.17.0, < 2.0) thor (>= 0.17.0, < 2.0)
middleman-core (4.3.5) middleman-core (4.3.11)
activesupport (>= 4.2, < 5.1) activesupport (>= 4.2, < 6.0)
addressable (~> 2.3) addressable (~> 2.3)
backports (~> 3.6) backports (~> 3.6)
bundler bundler
@@ -70,33 +71,34 @@ GEM
servolux servolux
tilt (~> 2.0.9) tilt (~> 2.0.9)
uglifier (~> 3.0) uglifier (~> 3.0)
minitest (5.12.2) minitest (5.14.3)
padrino-helpers (0.13.3.4) padrino-helpers (0.13.3.4)
i18n (~> 0.6, >= 0.6.7) i18n (~> 0.6, >= 0.6.7)
padrino-support (= 0.13.3.4) padrino-support (= 0.13.3.4)
tilt (>= 1.4.1, < 3) tilt (>= 1.4.1, < 3)
padrino-support (0.13.3.4) padrino-support (0.13.3.4)
activesupport (>= 3.1) activesupport (>= 3.1)
parallel (1.17.0) parallel (1.20.1)
public_suffix (4.0.1) public_suffix (4.0.6)
rack (2.0.7) rack (2.2.3)
rb-fsevent (0.10.3) rb-fsevent (0.10.4)
rb-inotify (0.10.0) rb-inotify (0.10.1)
ffi (~> 1.0) ffi (~> 1.0)
sassc (2.2.1) rexml (3.2.4)
sassc (2.4.0)
ffi (~> 1.9) ffi (~> 1.9)
servolux (0.13.0) servolux (0.13.0)
temple (0.8.2) temple (0.8.2)
thor (0.20.3) thor (1.1.0)
thread_safe (0.3.6) thread_safe (0.3.6)
tilt (2.0.10) tilt (2.0.10)
tzinfo (1.2.5) tzinfo (1.2.9)
thread_safe (~> 0.1) thread_safe (~> 0.1)
uglifier (3.2.0) uglifier (3.2.0)
execjs (>= 0.3.0, < 3) execjs (>= 0.3.0, < 3)
PLATFORMS PLATFORMS
ruby x86_64-linux
DEPENDENCIES DEPENDENCIES
middleman (~> 4.2) middleman (~> 4.2)
@@ -105,4 +107,4 @@ DEPENDENCIES
wdm (~> 0.1) wdm (~> 0.1)
BUNDLED WITH BUNDLED WITH
2.1.4 2.2.4

View File

@@ -1,2 +0,0 @@
README.md
yarn.lock

View File

@@ -1,2 +1 @@
README.md README.md
build

View File

@@ -13,6 +13,6 @@
"nuxt": "^2.0.0" "nuxt": "^2.0.0"
}, },
"devDependencies": { "devDependencies": {
"nodemon": "^1.18.9" "nodemon": "^2.0.7"
} }
} }

8116
examples/nuxtjs/yarn.lock Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -1,10 +0,0 @@
# editorconfig.org
root = true
[*]
charset = utf-8
end_of_line = lf
indent_size = 2
indent_style = space
insert_final_newline = true
trim_trailing_whitespace = true

View File

@@ -3,5 +3,11 @@
# system. Any custom values should go in .env and .env should *not* be checked # system. Any custom values should go in .env and .env should *not* be checked
# into version control. # into version control.
# schema.prisma defaults
DATABASE_URL=file:./dev.db DATABASE_URL=file:./dev.db
BINARY_TARGET=native
# location of the test database for api service scenarios (defaults to ./.redwood/test.db if not set)
# TEST_DATABASE_URL=file:./.redwood/test.db
# disables Prisma CLI update notifier
PRISMA_HIDE_UPDATE_MESSAGE=true

View File

@@ -1,10 +1,13 @@
.idea
.DS_Store .DS_Store
.env .env
.netlify .netlify
.redwood
dev.db dev.db
dist dist
dist-babel dist-babel
node_modules node_modules
yarn-error.log yarn-error.log
web/public/mockServiceWorker.js
.vercel .vercel

View File

@@ -1 +0,0 @@
lts/*

View File

@@ -5,11 +5,11 @@ datasource DS {
generator client { generator client {
provider = "prisma-client-js" provider = "prisma-client-js"
binaryTargets = env("BINARY_TARGET") binaryTargets = "native"
} }
// Define your own datamodels here and run `yarn redwood db save` to create // Define your own datamodels here and run `yarn redwood prisma migrate dev`
// migrations for them. // to create migrations for them and apply to your dev DB.
// TODO: Please remove the following example: // TODO: Please remove the following example:
model UserExample { model UserExample {
id Int @id @default(autoincrement()) id Int @id @default(autoincrement())

View File

@@ -6,6 +6,8 @@ dotenv.config()
const db = new PrismaClient() const db = new PrismaClient()
async function main() { async function main() {
// https://www.prisma.io/docs/guides/prisma-guides/seed-database
//
// Seed data is database data that needs to exist for your app to run. // Seed data is database data that needs to exist for your app to run.
// Ideally this file should be idempotent: running it multiple times // Ideally this file should be idempotent: running it multiple times
// will result in the same database state (usually by checking for the // will result in the same database state (usually by checking for the
@@ -16,11 +18,11 @@ async function main() {
// await db.user.create({ data: { name: 'Admin', email: 'admin@email.com' }}) // await db.user.create({ data: { name: 'Admin', email: 'admin@email.com' }})
// } // }
console.info('No data to seed. See api/prisma/seeds.js for info.') console.info('No data to seed. See api/db/seed.js for info.')
} }
main() main()
.catch((e) => console.error(e)) .catch((e) => console.error(e))
.finally(async () => { .finally(async () => {
await db.disconnect() await db.$disconnect()
}) })

View File

@@ -0,0 +1,6 @@
const { getConfig } = require('@redwoodjs/core')
const config = getConfig({ type: 'jest', target: 'node' })
config.displayName.name = 'api'
module.exports = config

View File

@@ -5,5 +5,5 @@
"src/*": ["./src/*"] "src/*": ["./src/*"]
} }
}, },
"include": ["src/**/*"] "include": ["src/**/*", "../.redwood/index.d.ts"]
} }

View File

@@ -3,6 +3,6 @@
"version": "0.0.0", "version": "0.0.0",
"private": true, "private": true,
"dependencies": { "dependencies": {
"@redwoodjs/api": "0.15.0" "@redwoodjs/api": "^0.25.0"
} }
} }

View File

@@ -3,17 +3,18 @@ import {
makeMergedSchema, makeMergedSchema,
makeServices, makeServices,
} from '@redwoodjs/api' } from '@redwoodjs/api'
import importAll from '@redwoodjs/api/importAll.macro'
import schemas from 'src/graphql/**/*.{js,ts}'
import { db } from 'src/lib/db' import { db } from 'src/lib/db'
import services from 'src/services/**/*.{js,ts}'
const schemas = importAll('api', 'graphql')
const services = importAll('api', 'services')
export const handler = createGraphQLHandler({ export const handler = createGraphQLHandler({
schema: makeMergedSchema({ schema: makeMergedSchema({
schemas, schemas,
services: makeServices({ services }), services: makeServices({ services }),
}), }),
db, onException: () => {
// Disconnect from your database with an unhandled exception.
db.$disconnect()
},
}) })

View File

@@ -1,4 +1,4 @@
// See https://github.com/prisma/prisma2/blob/master/docs/prisma-client-js/api.md#constructor // See https://www.prisma.io/docs/reference/tools-and-interfaces/prisma-client/constructor
// for options. // for options.
import { PrismaClient } from '@prisma/client' import { PrismaClient } from '@prisma/client'

View File

@@ -0,0 +1,7 @@
const { getConfig } = require('@redwoodjs/internal')
const config = getConfig()
module.exports = {
schema: `http://${config.api.host}:${config.api.port}/graphql`,
}

View File

@@ -3,11 +3,12 @@
"workspaces": { "workspaces": {
"packages": [ "packages": [
"api", "api",
"web" "web",
"packages/*"
] ]
}, },
"devDependencies": { "devDependencies": {
"@redwoodjs/core": "0.15.0" "@redwoodjs/core": "^0.25.0"
}, },
"eslintConfig": { "eslintConfig": {
"extends": "@redwoodjs/eslint-config" "extends": "@redwoodjs/eslint-config"

View File

@@ -6,4 +6,12 @@ module.exports = {
semi: false, semi: false,
singleQuote: true, singleQuote: true,
arrowParens: 'always', arrowParens: 'always',
overrides: [
{
files: 'Routes.js',
options: {
printWidth: 200,
},
},
],
} }

View File

@@ -1,7 +1,15 @@
# This file contains the configuration settings for your Redwood app.
# This file is also what makes your Redwood app a Redwood app.
# If you remove it and try to run `yarn rw dev`, you'll get an error.
#
# For the full list of options, see the "App Configuration: redwood.toml" doc:
# https://redwoodjs.com/docs/app-configuration-redwood-toml
[web] [web]
port = 8910 port = 8910
apiProxyPath = "/api" apiProxyPath = "/api"
[api] [api]
port = 8911 port = 8911
schemaPath = "./api/db/schema.prisma"
[browser] [browser]
open = false open = false

View File

@@ -0,0 +1,6 @@
const { getConfig } = require('@redwoodjs/core')
const config = getConfig({ type: 'jest', target: 'browser' })
config.displayName.name = 'web'
module.exports = config

View File

@@ -6,5 +6,5 @@
}, },
"jsx": "preserve" "jsx": "preserve"
}, },
"include": ["src/**/*"] "include": ["src/**/*", "../.redwood/index.d.ts"]
} }

View File

@@ -2,12 +2,20 @@
"name": "web", "name": "web",
"version": "0.0.0", "version": "0.0.0",
"private": true, "private": true,
"browserslist": [ "browserslist": {
"defaults" "development": [
], "last 1 version"
],
"production": [
"defaults",
"not IE 11",
"not IE_Mob 11"
]
},
"dependencies": { "dependencies": {
"@redwoodjs/router": "0.15.0", "@redwoodjs/forms": "^0.25.0",
"@redwoodjs/web": "0.15.0", "@redwoodjs/router": "^0.25.0",
"@redwoodjs/web": "^0.25.0",
"prop-types": "^15.7.2", "prop-types": "^15.7.2",
"react": "^16.13.1", "react": "^16.13.1",
"react-dom": "^16.13.1" "react-dom": "^16.13.1"

View File

@@ -1,13 +1,19 @@
# Static Assets # Static Assets
Use this folder to add static files directly to your app. All included files and folders will be copied directly into the `/dist` folder (created when Webpack builds for production). They will also be available during development when you run `yarn rw dev`. Use this folder to add static files directly to your app. All included files and folders will be copied directly into the `/dist` folder (created when Webpack builds for production). They will also be available during development when you run `yarn rw dev`.
>Note: files will *not* hot reload while the development server is running. You'll need to manually stop/start to access file changes.
> Note: files will _not_ hot reload while the development server is running. You'll need to manually stop/start to access file changes.
### Example Use ### Example Use
A file like `favicon.png` will be copied to `/dist/favicon.png`. A folder containing a file such as `static-files/my-logo.jpg` will be copied to `/dist/static-files/my-logo.jpg`. These can be referenced in your code directly without any special handling, e.g. A file like `favicon.png` will be copied to `/dist/favicon.png`. A folder containing a file such as `static-files/my-logo.jpg` will be copied to `/dist/static-files/my-logo.jpg`. These can be referenced in your code directly without any special handling, e.g.
``` ```
<link rel="icon" type="image/png" href="/favicon.png" /> <link rel="icon" type="image/png" href="/favicon.png" />
``` ```
and and
``` ```
<img src="/static-files/my-logo.jpg"> alt="Logo" /> <img src="/static-files/my-logo.jpg"> alt="Logo" />
``` ```
@@ -15,12 +21,15 @@ and
Behind the scenes, we are using Webpack's ["copy-webpack-plugin"](https://github.com/webpack-contrib/copy-webpack-plugin). Behind the scenes, we are using Webpack's ["copy-webpack-plugin"](https://github.com/webpack-contrib/copy-webpack-plugin).
## Best Practices ## Best Practices
Because assets in this folder are bypassing the javascript module system, **this folder should be used sparingly** for assets such as favicons, robots.txt, manifests, libraries incompatible with Webpack, etc. Because assets in this folder are bypassing the javascript module system, **this folder should be used sparingly** for assets such as favicons, robots.txt, manifests, libraries incompatible with Webpack, etc.
In general, it's best to import files directly into a template, page, or component. This allows Webpack to include that file in the bundle, which ensures Webpack will correctly process and move assets into the distribution folder, providing error checks and correct paths along the way. In general, it's best to import files directly into a template, page, or component. This allows Webpack to include that file in the bundle, which ensures Webpack will correctly process and move assets into the distribution folder, providing error checks and correct paths along the way.
### Example Asset Import with Webpack ### Example Asset Import with Webpack
Instead of handling our logo image as a static file per the example above, we can do the following: Instead of handling our logo image as a static file per the example above, we can do the following:
``` ```
import React from "react" import React from "react"
import logo from "./my-logo.jpg" import logo from "./my-logo.jpg"

View File

@@ -1,16 +1,18 @@
import ReactDOM from 'react-dom' import ReactDOM from 'react-dom'
import { RedwoodProvider, FatalErrorBoundary } from '@redwoodjs/web'
import FatalErrorPage from 'src/pages/FatalErrorPage'
import { FatalErrorBoundary } from '@redwoodjs/web'
import { RedwoodApolloProvider } from '@redwoodjs/web/apollo'
import FatalErrorPage from 'src/pages/FatalErrorPage'
import Routes from 'src/Routes' import Routes from 'src/Routes'
import './index.css' import './index.css'
ReactDOM.render( ReactDOM.render(
<FatalErrorBoundary page={FatalErrorPage}> <FatalErrorBoundary page={FatalErrorPage}>
<RedwoodProvider> <RedwoodApolloProvider>
<Routes /> <Routes />
</RedwoodProvider> </RedwoodApolloProvider>
</FatalErrorBoundary>, </FatalErrorBoundary>,
document.getElementById('redwood-app') document.getElementById('redwood-app')
) )

View File

@@ -1,4 +1,6 @@
export default () => ( import { Link, routes } from '@redwoodjs/router'
const AboutPage = () => (
<main> <main>
<style <style
dangerouslySetInnerHTML={{ dangerouslySetInnerHTML={{
@@ -36,9 +38,16 @@ export default () => (
}} }}
/> />
<section> <section>
<h1> <h1>About</h1>
<span>About RedwoodJS</span> <p>
</h1> Find me in <code>./web/src/pages/AboutPage/AboutPage.js</code>
</p>
<p>
My default route is named <code>about</code>, link to me with `
<Link to={routes.about()}>About</Link>`
</p>
</section> </section>
</main> </main>
) )
export default AboutPage

View File

@@ -1,4 +1,6 @@
export default () => ( import { Link, routes } from '@redwoodjs/router'
const HomePage = () => (
<main> <main>
<style <style
dangerouslySetInnerHTML={{ dangerouslySetInnerHTML={{
@@ -36,9 +38,16 @@ export default () => (
}} }}
/> />
<section> <section>
<h1> <h1>Welcome to RedwoodJS!</h1>
<span>Welcome to RedwoodJS!</span> <p>
</h1> Find me in <code>./web/src/pages/HomePage/HomePage.js</code>
</p>
<p>
My default route is named <code>home</code>, link to me with `
<Link to={routes.home()}>Home</Link>`
</p>
</section> </section>
</main> </main>
) )
export default HomePage

View File

@@ -0,0 +1,7 @@
import HomePage from './HomePage'
export const generated = () => {
return <HomePage />
}
export default { title: 'Pages/HomePage' }

File diff suppressed because it is too large Load Diff

2
examples/zola/.gitignore vendored Normal file
View File

@@ -0,0 +1,2 @@
/public
/.vercel

16
examples/zola/config.toml Normal file
View File

@@ -0,0 +1,16 @@
# The URL the site will be built for
base_url = "/"
# Whether to automatically compile all Sass files in the sass directory
compile_sass = true
# Whether to build a search index to be used later on by a JavaScript library
build_search_index = false
[markdown]
# Whether to do syntax highlighting
# Theme can be customised by setting the `highlight_theme` variable to a theme supported by Zola
highlight_code = false
[extra]
# Put all your custom variables here

View File

@@ -0,0 +1,6 @@
+++
title = "List of blog posts"
sort_by = "date"
template = "blog.html"
page_template = "blog-page.html"
+++

View File

@@ -0,0 +1,6 @@
+++
title = "My first post"
date = 2019-11-27
+++
This is my first blog post.

View File

@@ -0,0 +1,6 @@
+++
title = "My second post"
date = 2019-11-28
+++
This is my second blog post.

View File

@@ -0,0 +1,18 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>MyBlog</title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bulma@0.8.0/css/bulma.min.css">
</head>
<body>
<section class="section">
<div class="container">
{% block content %} {% endblock %}
</div>
</section>
</body>
</html>

View File

@@ -0,0 +1,9 @@
{% extends "base.html" %}
{% block content %}
<h1 class="title">
{{ page.title }}
</h1>
<p class="subtitle"><strong>{{ page.date }}</strong></p>
{{ page.content | safe }}
{% endblock content %}

View File

@@ -0,0 +1,12 @@
{% extends "base.html" %}
{% block content %}
<h1 class="title">
{{ section.title }}
</h1>
<ul>
{% for page in section.pages %}
<li><a href="{{ page.permalink | safe }}">{{ page.title }}</a></li>
{% endfor %}
</ul>
{% endblock content %}

View File

@@ -0,0 +1,8 @@
{% extends "base.html" %}
{% block content %}
<h1 class="title">
This is my blog made with Zola.
</h1>
<p>Click <a href="/blog/">here</a> to see my posts.</p>
{% endblock content %}

View File

@@ -39,7 +39,7 @@
"publish-from-github": "./utils/publish.sh", "publish-from-github": "./utils/publish.sh",
"changelog": "node utils/changelog.js", "changelog": "node utils/changelog.js",
"build": "node utils/run.js build all", "build": "node utils/run.js build all",
"now-build": "mkdir -p public && echo '<a href=\"https://vercel.com/import\">Import</a>' > public/output.html", "vercel-build": "mkdir -p public && echo '<a href=\"https://vercel.com/import\">Import</a>' > public/output.html",
"test-unit": "node utils/run.js test-unit", "test-unit": "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",

View File

@@ -1,945 +0,0 @@
[
{
"name": "Blitz.js",
"slug": "blitzjs",
"demo": "https://blitzjs.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/blitz.svg",
"tagline": "Blitz.js: The Fullstack React Framework",
"description": "A brand new Blitz.js app - the result of running `npx blitz new`.",
"website": "https://blitzjs.com",
"useRuntime": { "src": "package.json", "use": "@vercel/next" },
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"blitz\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `blitz build`"
},
"devCommand": {
"value": "blitz start"
},
"outputDirectory": {
"placeholder": "Next.js default"
}
}
},
{
"name": "Next.js",
"slug": "nextjs",
"demo": "https://nextjs.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/next.svg",
"tagline": "Next.js makes you productive with React instantly — whether you want to build static or dynamic sites.",
"description": "A Next.js app and a Serverless Function API.",
"website": "https://nextjs.org",
"sort": 1,
"useRuntime": { "src": "package.json", "use": "@vercel/next" },
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"next\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `next build`"
},
"devCommand": {
"value": "next dev --port $PORT"
},
"outputDirectory": {
"placeholder": "Next.js default"
}
},
"recommendedIntegrations": [
{
"id": "oac_5lUsiANun1DEzgLg0NZx5Es3",
"dependencies": ["next-plugin-sentry", "next-sentry-source-maps"]
}
]
},
{
"name": "Gatsby.js",
"slug": "gatsby",
"demo": "https://gatsby.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/gatsby.svg",
"tagline": "Gatsby helps developers build blazing fast websites and apps with React.",
"description": "A Gatsby app, using the default starter theme and a Serverless Function API.",
"website": "https://gatsbyjs.org",
"sort": 2,
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"gatsby\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `gatsby build`"
},
"devCommand": {
"value": "gatsby develop --port $PORT"
},
"outputDirectory": {
"value": "public"
}
}
},
{
"name": "Hexo",
"slug": "hexo",
"demo": "https://hexo.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/hexo.svg",
"tagline": "Hexo is a fast, simple & powerful blog framework powered by Node.js.",
"description": "A Hexo site, created with the Hexo CLI.",
"website": "https://hexo.io",
"sort": 3,
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"hexo\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `hexo generate`"
},
"devCommand": {
"value": "hexo server --port $PORT"
},
"outputDirectory": {
"value": "public"
}
}
},
{
"name": "Eleventy",
"slug": "eleventy",
"demo": "https://eleventy.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/eleventy.svg",
"tagline": "11ty is a simpler static site generator written in JavaScript, created to be an alternative to Jekyll.",
"description": "An Eleventy site, created with npm init.",
"website": "https://www.11ty.dev",
"sort": 4,
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@11ty\\/eleventy\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `npx @11ty/eleventy`"
},
"devCommand": {
"value": "npx @11ty/eleventy --serve --watch --port $PORT"
},
"outputDirectory": {
"value": "_site"
}
}
},
{
"name": "Docusaurus 2",
"slug": "docusaurus-2",
"demo": "https://docusaurus-2.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/docusaurus.svg",
"tagline": "Docusaurus makes it easy to maintain Open Source documentation websites.",
"description": "A static Docusaurus site that makes it easy to maintain OSS documentation.",
"website": "https://v2.docusaurus.io",
"detectors": {
"some": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@docusaurus\\/core\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `docusaurus build`"
},
"devCommand": {
"value": "docusaurus start --port $PORT"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Docusaurus 1",
"slug": "docusaurus",
"demo": "https://docusaurus.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/docusaurus.svg",
"tagline": "Docusaurus makes it easy to maintain Open Source documentation websites.",
"description": "A static Docusaurus site that makes it easy to maintain OSS documentation.",
"website": "https://docusaurus.io/",
"detectors": {
"some": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"docusaurus\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `docusaurus-build`"
},
"devCommand": {
"value": "docusaurus-start --port $PORT"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Preact",
"slug": "preact",
"demo": "https://preact.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/preact.svg",
"tagline": "Preact is a fast 3kB alternative to React with the same modern API.",
"description": "A Preact app, created with the Preact CLI.",
"website": "https://preactjs.com",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"preact-cli\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `preact build`"
},
"devCommand": {
"value": "preact watch --port $PORT"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Dojo",
"slug": "dojo",
"demo": "https://dojo.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/dojo.png",
"tagline": "Dojo is a modern progressive, TypeScript first framework.",
"description": "A Dojo app, created with the Dojo CLI's cli-create-app command.",
"website": "https://dojo.io",
"detectors": {
"some": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@dojo\\/framework\":\\s*\".+?\"[^}]*}"
},
{
"path": ".dojorc"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `dojo build`"
},
"devCommand": {
"value": "dojo build -m dev -w -s -p $PORT"
},
"outputDirectory": {
"value": "output/dist"
}
}
},
{
"name": "Ember.js",
"slug": "ember",
"demo": "https://ember.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/ember.svg",
"tagline": "Ember.js helps webapp developers be more productive out of the box.",
"description": "An Ember app, created with the Ember CLI.",
"website": "https://emberjs.com/",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"ember-cli\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `ember build`"
},
"devCommand": {
"value": "ember serve --port $PORT"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "Vue.js",
"slug": "vue",
"demo": "https://vue.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/vue.svg",
"tagline": "Vue.js is a versatile JavaScript framework that is as approachable as it is performant.",
"description": "A Vue.js app, created with the Vue CLI.",
"website": "https://vuejs.org",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@vue\\/cli-service\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `vue-cli-service build`"
},
"devCommand": {
"value": "vue-cli-service serve --port $PORT"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "Scully",
"slug": "scully",
"demo": "https://scully.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/scullyio-logo.png",
"tagline": "Scully is a static site generator for Angular.",
"description": "The Static Site Generator for Angular apps.",
"website": "https://github.com/scullyio/scully",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@scullyio\\/init\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `ng build && scully`"
},
"devCommand": {
"value": "ng serve --port $PORT"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "Ionic Angular",
"slug": "ionic-angular",
"demo": "https://ionic-angular.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/ionic.svg",
"tagline": "Ionic Angular allows you to build mobile PWAs with Angular and the Ionic Framework.",
"description": "An Ionic Angular site, created with the Ionic CLI.",
"website": "https://ionicframework.com",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@ionic\\/angular\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `ng build`"
},
"devCommand": {
"value": "ng start"
},
"outputDirectory": {
"value": "www"
}
}
},
{
"name": "Angular",
"slug": "angular",
"demo": "https://angular.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/angular.svg",
"tagline": "Angular is a TypeScript-based cross-platform framework from Google.",
"description": "An Angular app, created with the Angular CLI.",
"website": "https://angular.io",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@angular\\/cli\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `ng build`"
},
"devCommand": {
"value": "ng serve --port $PORT"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "Polymer",
"slug": "polymer",
"demo": "https://polymer.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/polymer.svg",
"tagline": "Polymer is an open-source webapps library from Google, for building using Web Components.",
"description": "A Polymer app, created with the Polymer CLI.",
"website": "https://www.polymer-project.org/",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"polymer-cli\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `polymer build`"
},
"devCommand": {
"value": "polymer serve --port $PORT"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Svelte",
"slug": "svelte",
"demo": "https://svelte.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/svelte.svg",
"tagline": "Svelte lets you write high performance reactive apps with significantly less boilerplate. ",
"description": "A Svelte app, using the Svelte template, and a Serverless Function API.",
"website": "https://svelte.dev",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"sirv-cli\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `rollup -c`"
},
"devCommand": {
"value": "sirv public --single --dev --port $PORT"
},
"outputDirectory": {
"value": "public"
}
}
},
{
"name": "Ionic React",
"slug": "ionic-react",
"demo": "https://ionic-react.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/ionic.svg",
"tagline": "Ionic React allows you to build mobile PWAs with React and the Ionic Framework.",
"description": "An Ionic React site, created with the Ionic CLI.",
"website": "https://ionicframework.com",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@ionic\\/react\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `react-scripts build`"
},
"devCommand": {
"value": "react-scripts start"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Create React App",
"slug": "create-react-app",
"demo": "https://react-functions.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/react.svg",
"tagline": "Create React App allows you to get going with React in no time.",
"description": "A React app, bootstrapped with create-react-app, and a Serverless Function API.",
"website": "https://create-react-app.dev",
"detectors": {
"some": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"react-scripts\":\\s*\".+?\"[^}]*}"
},
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"react-dev-utils\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `react-scripts build`"
},
"devCommand": {
"value": "react-scripts start"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Gridsome",
"slug": "gridsome",
"demo": "https://gridsome.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/gridsome.svg",
"tagline": "Gridsome is a Vue.js-powered framework for building websites & apps that are fast by default.",
"description": "A Gridsome app, created with the Gridsome CLI.",
"website": "https://gridsome.org/",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"gridsome\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `gridsome build`"
},
"devCommand": {
"value": "gridsome develop -p $PORT"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "UmiJS",
"slug": "umijs",
"demo": "https://umijs.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/umi.svg",
"tagline": "UmiJS is an extensible enterprise-level React application framework.",
"description": "An UmiJS app, created using the Umi CLI.",
"website": "https://umijs.org",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"umi\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `umi build`"
},
"devCommand": {
"value": "umi dev --port $PORT"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "Sapper",
"slug": "sapper",
"demo": "https://sapper.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/svelte.svg",
"tagline": "Sapper is a framework for building high-performance universal web apps with Svelte.",
"description": "A Sapper app, using the Sapper template.",
"website": "https://sapper.svelte.dev",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"sapper\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `sapper export`"
},
"devCommand": {
"value": "sapper dev --port $PORT"
},
"outputDirectory": {
"value": "__sapper__/export"
}
}
},
{
"name": "Saber",
"slug": "saber",
"demo": "https://saber.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/saber.svg",
"tagline": "Saber is a framework for building static sites in Vue.js that supports data from any source.",
"description": "A Saber site, created with npm init.",
"website": "https://saber.land/",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"saber\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `saber build`"
},
"devCommand": {
"value": "saber --port $PORT"
},
"outputDirectory": {
"value": "public"
}
}
},
{
"name": "Stencil",
"slug": "stencil",
"demo": "https://stencil.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/stencil.svg",
"tagline": "Stencil is a powerful toolchain for building Progressive Web Apps and Design Systems.",
"description": "A Stencil site, created with the Stencil CLI.",
"website": "https://stenciljs.com/",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@stencil\\/core\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `stencil build`"
},
"devCommand": {
"value": "stencil build --dev --watch --serve --port $PORT"
},
"outputDirectory": {
"value": "www"
}
}
},
{
"name": "Nuxt.js",
"slug": "nuxtjs",
"demo": "https://nuxtjs.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/nuxt.svg",
"tagline": "Nuxt.js is the web comprehensive framework that lets you dream big with Vue.js.",
"description": "A Nuxt.js app, bootstrapped with create-nuxt-app.",
"website": "https://nuxtjs.org",
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"nuxt(-edge)?\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `nuxt generate`"
},
"devCommand": {
"value": "nuxt"
},
"outputDirectory": {
"value": "dist"
}
}
},
{
"name": "RedwoodJS",
"slug": "redwoodjs",
"demo": "https://redwoodjs.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/redwoodjs.svg",
"tagline": "RedwoodJS is a full-stack framework for the Jamstack.",
"description": "A RedwoodJS app, bootstraped with create-redwood-app.",
"website": "https://redwoodjs.com",
"useRuntime": { "src": "package.json", "use": "@vercel/redwood" },
"ignoreRuntimes": ["@vercel/node"],
"detectors": {
"every": [
{
"path": "package.json",
"matchContent": "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"@redwoodjs\\/core\":\\s*\".+?\"[^}]*}"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"value": "yarn rw build && yarn rw db up --no-db-client --auto-approve && yarn rw dataMigrate up"
},
"devCommand": {
"value": "yarn rw dev --fwd=\"--port=$PORT --open=false\""
},
"outputDirectory": {
"placeholder": "RedwoodJS default"
}
}
},
{
"name": "Hugo",
"slug": "hugo",
"demo": "https://hugo.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/hugo.svg",
"tagline": "Hugo is the worlds fastest framework for building websites, written in Go.",
"description": "A Hugo site, created with the Hugo CLI.",
"website": "https://gohugo.io",
"sort": 5,
"detectors": {
"some": [
{
"path": "config.yaml"
},
{
"path": "config.toml"
},
{
"path": "config.json"
}
]
},
"settings": {
"installCommand": {
"placeholder": "None"
},
"buildCommand": {
"placeholder": "`npm run build` or `hugo -D --gc`"
},
"devCommand": {
"value": "hugo server -D -w -p $PORT"
},
"outputDirectory": {
"placeholder": "`public` or `publishDir` from the `config` file"
}
}
},
{
"name": "Jekyll",
"slug": "jekyll",
"demo": "https://jekyll.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/jekyll.svg",
"tagline": "Jekyll makes it super easy to transform your plain text into static websites and blogs.",
"description": "A Jekyll site, created with the Jekyll CLI.",
"website": "https://jekyllrb.com/",
"detectors": {
"every": [
{
"path": "_config.yml"
}
]
},
"settings": {
"installCommand": {
"value": "bundle install"
},
"buildCommand": {
"placeholder": "`npm run build` or `jekyll build`"
},
"devCommand": {
"value": "bundle exec jekyll serve --watch --port $PORT"
},
"outputDirectory": {
"placeholder": "`_site` or `destination` from `_config.yml`"
}
}
},
{
"name": "Brunch",
"slug": "brunch",
"demo": "https://brunch.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/brunch.svg",
"tagline": "Brunch is a fast and simple webapp build tool with seamless incremental compilation for rapid development.",
"description": "A Brunch app, created with the Brunch CLI.",
"website": "https://brunch.io/",
"detectors": {
"every": [
{
"path": "brunch-config.js"
}
]
},
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run build` or `brunch build --production`"
},
"devCommand": {
"value": "brunch watch --server --port $PORT"
},
"outputDirectory": {
"value": "public"
}
}
},
{
"name": "Middleman",
"slug": "middleman",
"demo": "https://middleman.now-examples.now.sh",
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/middleman.svg",
"tagline": "Middleman is a static site generator that uses all the shortcuts and tools in modern web development.",
"description": "A Middleman app, created with the Middleman CLI.",
"website": "https://middlemanapp.com/",
"detectors": {
"every": [
{
"path": "config.rb"
}
]
},
"settings": {
"installCommand": {
"value": "bundle install"
},
"buildCommand": {
"value": "`npm run build` or `bundle exec middleman build`"
},
"devCommand": {
"value": "bundle exec middleman server -p $PORT"
},
"outputDirectory": {
"value": "build"
}
}
},
{
"name": "Other",
"slug": null,
"logo": "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/other.svg",
"description": "No framework or a unoptimized framework.",
"settings": {
"installCommand": {
"placeholder": "`yarn install` or `npm install`"
},
"buildCommand": {
"placeholder": "`npm run vercel-build` or `npm run build`"
},
"devCommand": {
"placeholder": "None"
},
"outputDirectory": {
"placeholder": "`public` if it exists, or `.`"
}
}
}
]

View File

@@ -1,38 +0,0 @@
export interface FrameworkDetectionItem {
path: string;
matchContent?: string;
}
export interface SettingPlaceholder {
placeholder: string;
}
export interface SettingValue {
value: string;
}
export type Setting = SettingValue | SettingPlaceholder;
export interface Framework {
name: string;
slug: string | null;
logo: string;
demo?: string;
tagline?: string;
website?: string;
description: string;
sort?: number;
useRuntime?: { src: string; use: string };
ignoreRuntimes?: string[];
detectors?: {
every?: FrameworkDetectionItem[];
some?: FrameworkDetectionItem[];
};
settings: {
buildCommand: Setting;
devCommand: Setting;
outputDirectory: Setting;
};
recommendedIntegrations?: {
id: string;
dependencies: string[];
}[];
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.1 KiB

View File

@@ -1,14 +1,26 @@
{ {
"name": "@vercel/frameworks", "name": "@vercel/frameworks",
"version": "0.2.0", "version": "0.3.0",
"main": "frameworks.json", "main": "./dist/frameworks.js",
"types": "./dist/frameworks.d.ts",
"files": [
"dist"
],
"license": "UNLICENSED", "license": "UNLICENSED",
"scripts": { "scripts": {
"build": "tsc",
"test-unit": "jest --env node --verbose --runInBand --bail" "test-unit": "jest --env node --verbose --runInBand --bail"
}, },
"dependencies": {
"@iarna/toml": "2.2.3",
"js-yaml": "3.13.1"
},
"devDependencies": { "devDependencies": {
"@types/jest": "24.0.22", "@types/jest": "24.0.22",
"@types/js-yaml": "3.12.1",
"@types/node": "12.0.4", "@types/node": "12.0.4",
"@types/node-fetch": "2.5.8",
"@vercel/routing-utils": "1.10.0",
"ajv": "6.12.2", "ajv": "6.12.2",
"jest": "24.9.0", "jest": "24.9.0",
"ts-jest": "24.1.0", "ts-jest": "24.1.0",

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,41 @@
import yaml from 'js-yaml';
import toml from '@iarna/toml';
import { promises } from 'fs';
const { readFile } = promises;
async function readFileOrNull(file: string) {
try {
const data = await readFile(file);
return data;
} catch (err) {
if (err.code !== 'ENOENT') {
throw err;
}
}
return null;
}
export async function readConfigFile<T>(
files: string | string[]
): Promise<T | null> {
files = Array.isArray(files) ? files : [files];
for (const name of files) {
const data = await readFileOrNull(name);
if (data) {
const str = data.toString('utf8');
if (name.endsWith('.json')) {
return JSON.parse(str) as T;
} else if (name.endsWith('.toml')) {
return (toml.parse(str) as unknown) as T;
} else if (name.endsWith('.yaml') || name.endsWith('.yml')) {
return yaml.safeLoad(str, { filename: name }) as T;
}
}
}
return null;
}

View File

@@ -0,0 +1,179 @@
import { Route } from '@vercel/routing-utils';
export interface FrameworkDetectionItem {
/**
* A file path
* @example "package.json"
*/
path: string;
/**
* A matcher
* @example "\"(dev)?(d|D)ependencies\":\\s*{[^}]*\"next\":\\s*\".+?\"[^}]*}"
*/
matchContent?: string;
}
export interface SettingPlaceholder {
/**
* A placeholder value for when the framework has not a predefined one
* @example "`npm run build` or `next build`"
*/
placeholder: string;
}
export interface SettingValue {
/**
* A predefined setting for the detected framework
* @example "next dev --port $PORT"
*/
value: string;
}
export type Setting = SettingValue | SettingPlaceholder;
/**
* Framework detection information.
*/
export interface Framework {
/**
* Name of the framework
* @example "Next.js"
*/
name: string;
/**
* A unique identifier for the framework
* @example "nextjs"
*/
slug: string | null;
/**
* A URL to the logo of the framework
* @example "https://raw.githubusercontent.com/vercel/vercel/master/packages/frameworks/logos/next.svg"
*/
logo: string;
/**
* A URL to a deployed example of the framework
* @example "https://nextjs.now-examples.vercel.app"
*/
demo?: string;
/**
* A marketing tagline for the framework
* @example "Next.js makes you productive with React instantly — whether you want to build static or dynamic sites."
*/
tagline?: string;
/**
* A URL to the official website of the framework
* @example "https://nextjs.org"
*/
website?: string;
/**
* Short description of the framework
* @example "A Next.js app and a Serverless Function API."
*/
description: string;
/**
* A ponderated value to sort matching frameworks
* @example 1
*/
sort?: number;
/**
* Runtime configuration required to run the framework in Vercel
*/
useRuntime?: {
/**
* Runtime source
* @example "package.json"
*/
src: string;
/**
* Runtime
* @example "@vercel/next"
*/
use: string;
};
ignoreRuntimes?: string[];
/**
* Detectors used to find out the framework
*/
detectors?: {
/**
* Collection of detectors that must be matched for the framework
* to be detected.
*/
every?: FrameworkDetectionItem[];
/**
* Collection of detectors where one match triggers the framework
* to be detected.
*/
some?: FrameworkDetectionItem[];
};
settings: {
/**
* Default Install Command or a placeholder
*/
installCommand: Setting;
/**
* Default Build Command or a placeholder
*/
buildCommand: Setting;
/**
* Default Development Command or a placeholder
*/
devCommand: Setting;
/**
* Default Output Directory
*/
outputDirectory: Setting;
};
/**
* A list of recommended integrations for the framework
*/
recommendedIntegrations?: {
/**
* Id of the recommended integration
* @example "oac_5lUsiANun1DEzgLg0NZx5Es3"
*/
id: string;
/**
* Dependencies of the recommended integration
* @example ["next-plugin-sentry", "next-sentry-source-maps"]
*/
dependencies: string[];
}[];
/**
* Name of a dependency in `package.json` to detect this framework.
* @example "hexo"
*/
dependency?: string;
/**
* Function that returns the name of the directory that the framework outputs
* its build results to. In some cases this is read from a configuration file.
*/
getOutputDirName: (dirPrefix: string) => Promise<string>;
/**
* An array (or a function that returns an array) of default `Route` rules that
* the framework uses.
* @example [{ handle: 'filesystem' }, { src: '.*', status: 404, dest: '404.html' }]
*/
defaultRoutes?: Route[] | ((dirPrefix: string) => Promise<Route[]>);
/**
* A glob string of files to cache for future deployments.
* @example ".cache/**"
*/
cachePattern?: string;
/**
* The default build command for the framework.
* @example "next build"
*/
buildCommand: string | null;
/**
* The default development command for the framework.
* @example "next dev"
*/
devCommand: string | null;
/**
* The default version of the framework command that is available within the
* build image. Usually an environment variable can be set to override this.
* @example "0.13.0"
*/
defaultVersion?: string;
}

View File

@@ -1,9 +1,11 @@
import Ajv from 'ajv'; import Ajv from 'ajv';
import assert from 'assert';
import { join } from 'path'; import { join } from 'path';
import { existsSync } from 'fs'; import { existsSync } from 'fs';
import { isString } from 'util'; import { isString } from 'util';
import { Framework } from '../'; import fetch from 'node-fetch';
const frameworkList = require('../frameworks.json') as Framework[]; import { URL, URLSearchParams } from 'url';
import frameworkList from '../src/frameworks';
const SchemaFrameworkDetectionItem = { const SchemaFrameworkDetectionItem = {
type: 'array', type: 'array',
@@ -53,8 +55,15 @@ const Schema = {
type: 'array', type: 'array',
items: { items: {
type: 'object', type: 'object',
required: ['name', 'slug', 'logo', 'description', 'settings'], required: [
additionalProperties: false, 'name',
'slug',
'logo',
'description',
'settings',
'buildCommand',
'devCommand',
],
properties: { properties: {
name: { type: 'string' }, name: { type: 'string' },
slug: { type: ['string', 'null'] }, slug: { type: ['string', 'null'] },
@@ -122,10 +131,25 @@ const Schema = {
}, },
}, },
}, },
dependency: { type: 'string' },
cachePattern: { type: 'string' },
buildCommand: { type: ['string', 'null'] },
devCommand: { type: ['string', 'null'] },
}, },
}, },
}; };
async function getDeployment(host: string) {
const query = new URLSearchParams();
query.set('url', host);
const res = await fetch(
`https://api.vercel.com/v11/deployments/get?${query}`
);
const body = await res.json();
return body;
}
describe('frameworks', () => { describe('frameworks', () => {
it('ensure there is an example for every framework', async () => { it('ensure there is an example for every framework', async () => {
const root = join(__dirname, '..', '..', '..'); const root = join(__dirname, '..', '..', '..');
@@ -173,4 +197,30 @@ describe('frameworks', () => {
} }
}); });
}); });
it('ensure unique slug', async () => {
const slugs = new Set<string>();
for (const { slug } of frameworkList) {
if (typeof slug === 'string') {
assert(!slugs.has(slug), `Slug "${slug}" is not unique`);
slugs.add(slug);
}
}
});
it('ensure all demo URLs are "public"', async () => {
await Promise.all(
frameworkList
.filter(f => typeof f.demo === 'string')
.map(async f => {
const url = new URL(f.demo!);
const deployment = await getDeployment(url.hostname);
assert.equal(
deployment.public,
true,
`Demo URL ${f.demo} is not "public"`
);
})
);
});
}); });

View File

@@ -15,6 +15,6 @@
"strict": true, "strict": true,
"target": "esnext" "target": "esnext"
}, },
"include": ["src/**/*", "test/**/*"], "include": ["src/*.ts"],
"exclude": ["node_modules"] "exclude": ["node_modules"]
} }

View File

@@ -0,0 +1,30 @@
#!/usr/bin/env node
const fs = require('fs-extra');
const execa = require('execa');
const { join } = require('path');
async function main() {
const outDir = join(__dirname, 'dist');
// Start fresh
await fs.remove(outDir);
// Compile TypeScript
await execa('tsc', [], { stdio: 'inherit' });
// Run `ncc`
const mainDir = join(outDir, 'main');
await execa('ncc', ['build', 'src/index.ts', '-o', mainDir], {
stdio: 'inherit',
});
// Move compiled ncc file to out dir
await fs.rename(join(mainDir, 'index.js'), join(outDir, 'index.js'));
// Delete leftover "main" dir
await fs.remove(mainDir);
}
main().catch(err => {
console.error(err);
process.exit(1);
});

View File

@@ -1,13 +0,0 @@
#!/bin/bash
set -euo pipefail
out="dist"
rm -rf "$out"
tsc
rm "$out/index.js"
ncc build "src/index.ts" -o "$out/main"
mv "$out/main/index.js" "$out/index.js"
rm -rf "$out/main"

View File

@@ -1,6 +1,6 @@
{ {
"name": "@vercel/build-utils", "name": "@vercel/build-utils",
"version": "2.7.0", "version": "2.10.0",
"license": "MIT", "license": "MIT",
"main": "./dist/index.js", "main": "./dist/index.js",
"types": "./dist/index.d.js", "types": "./dist/index.d.js",
@@ -11,10 +11,10 @@
"directory": "packages/now-build-utils" "directory": "packages/now-build-utils"
}, },
"scripts": { "scripts": {
"build": "./build.sh", "build": "node build",
"test-unit": "jest --env node --verbose --runInBand --bail test/unit.*test.*", "test-unit": "jest --env node --verbose --runInBand --bail test/unit.*test.*",
"test-integration-once": "jest --env node --verbose --runInBand --bail test/integration.test.js", "test-integration-once": "jest --env node --verbose --runInBand --bail test/integration.test.js",
"prepublishOnly": "./build.sh" "prepublishOnly": "node build"
}, },
"devDependencies": { "devDependencies": {
"@iarna/toml": "2.2.3", "@iarna/toml": "2.2.3",
@@ -29,7 +29,7 @@
"@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.1",
"@vercel/frameworks": "0.2.0", "@vercel/frameworks": "0.3.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",

View File

@@ -2,10 +2,9 @@ import minimatch from 'minimatch';
import { valid as validSemver } from 'semver'; import { valid as validSemver } from 'semver';
import { parse as parsePath, extname } from 'path'; import { parse as parsePath, extname } from 'path';
import { Route, Source } from '@vercel/routing-utils'; import { Route, Source } from '@vercel/routing-utils';
import _frameworks, { Framework } from '@vercel/frameworks'; import frameworkList, { Framework } from '@vercel/frameworks';
import { PackageJson, Builder, Config, BuilderFunctions } from './types'; import { PackageJson, Builder, Config, BuilderFunctions } from './types';
import { isOfficialRuntime } from './'; import { isOfficialRuntime } from './';
const frameworkList = _frameworks as Framework[];
const slugToFramework = new Map<string | null, Framework>( const slugToFramework = new Map<string | null, Framework>(
frameworkList.map(f => [f.slug, f]) frameworkList.map(f => [f.slug, f])
); );
@@ -312,11 +311,16 @@ export async function detectBuilders(
if (frontendBuilder) { if (frontendBuilder) {
builders.push(frontendBuilder); builders.push(frontendBuilder);
if (hasNextApiFiles && apiBuilders.length) { if (
hasNextApiFiles &&
apiBuilders.some(b => isOfficialRuntime('node', b.use))
) {
warnings.push({ warnings.push({
code: 'conflicting_files', code: 'conflicting_files',
message: message:
'It is not possible to use `api` and `pages/api` at the same time, please only use one option', 'When using Next.js, it is recommended to place Node.js Serverless Functions inside of the `pages/api` (provided by Next.js) directory instead of `api` (provided by Vercel).',
link: 'https://nextjs.org/docs/api-routes/introduction',
action: 'Learn More',
}); });
} }
} }
@@ -945,8 +949,8 @@ function getRouteResult(
const rewriteRoutes: Route[] = []; const rewriteRoutes: Route[] = [];
const errorRoutes: Route[] = []; const errorRoutes: Route[] = [];
const framework = frontendBuilder?.config?.framework || ''; const framework = frontendBuilder?.config?.framework || '';
const use = frontendBuilder?.use || ''; const isNextjs =
const isNextjs = framework === 'nextjs' || use.startsWith('@vercel/next'); framework === 'nextjs' || isOfficialRuntime('next', frontendBuilder?.use);
const ignoreRuntimes = slugToFramework.get(framework)?.ignoreRuntimes; const ignoreRuntimes = slugToFramework.get(framework)?.ignoreRuntimes;
if (apiRoutes && apiRoutes.length > 0) { if (apiRoutes && apiRoutes.length > 0) {

View File

@@ -3,7 +3,7 @@ import { DetectorFilesystem } from './detectors/filesystem';
export interface DetectFrameworkOptions { export interface DetectFrameworkOptions {
fs: DetectorFilesystem; fs: DetectorFilesystem;
frameworkList: Framework[]; frameworkList: readonly Framework[];
} }
async function matches(fs: DetectorFilesystem, framework: Framework) { async function matches(fs: DetectorFilesystem, framework: Framework) {

View File

@@ -51,8 +51,8 @@ export default async function glob(
stat, stat,
`statCache does not contain value for ${relativePath} (resolved to ${fsPath})` `statCache does not contain value for ${relativePath} (resolved to ${fsPath})`
); );
if (stat.isFile()) { const isSymlink = options.symlinks![fsPath];
const isSymlink = options.symlinks![fsPath]; if (isSymlink || stat.isFile()) {
if (isSymlink) { if (isSymlink) {
stat = await lstat(fsPath); stat = await lstat(fsPath);
} }

View File

@@ -1,31 +1,36 @@
import { intersects, validRange } from 'semver'; import { intersects, validRange } from 'semver';
import boxen from 'boxen';
import { NodeVersion } from '../types'; import { NodeVersion } from '../types';
import { NowBuildError } from '../errors'; import { NowBuildError } from '../errors';
import debug from '../debug'; import debug from '../debug';
const allOptions: NodeVersion[] = [ const allOptions = [
{ major: 14, range: '14.x', runtime: 'nodejs14.x' },
{ major: 12, range: '12.x', runtime: 'nodejs12.x' }, { major: 12, range: '12.x', runtime: 'nodejs12.x' },
{ major: 10, range: '10.x', runtime: 'nodejs10.x' }, {
major: 10,
range: '10.x',
runtime: 'nodejs10.x',
discontinueDate: new Date('2021-04-20'),
},
{ {
major: 8, major: 8,
range: '8.10.x', range: '8.10.x',
runtime: 'nodejs8.10', runtime: 'nodejs8.10',
discontinueDate: new Date('2020-01-06'), discontinueDate: new Date('2020-01-06'),
}, },
]; ] as const;
function getHint(isAuto: boolean) {
const { major, range } = getLatestNodeVersion();
return isAuto
? `Please set Node.js Version to ${range} in your Project Settings to use Node.js ${major}.`
: `Please set "engines": { "node": "${range}" } in your \`package.json\` file to use Node.js ${major}.`;
}
const pleaseSet =
'Please set "engines": { "node": "' +
getLatestNodeVersion().range +
'" } in your `package.json` file to upgrade to Node.js ' +
getLatestNodeVersion().major +
'.';
const upstreamProvider = const upstreamProvider =
'This change is the result of a decision made by an upstream infrastructure provider (AWS).' + 'This change is the result of a decision made by an upstream infrastructure provider (AWS).';
'\nRead more: https://docs.aws.amazon.com/lambda/latest/dg/runtime-support-policy.html';
export function getLatestNodeVersion(): NodeVersion { export function getLatestNodeVersion() {
return allOptions[0]; return allOptions[0];
} }
@@ -34,10 +39,10 @@ export function getDiscontinuedNodeVersions(): NodeVersion[] {
} }
export async function getSupportedNodeVersion( export async function getSupportedNodeVersion(
engineRange?: string, engineRange: string | undefined,
isAuto?: boolean isAuto: boolean
): Promise<NodeVersion> { ): Promise<NodeVersion> {
let selection = getLatestNodeVersion(); let selection: NodeVersion = getLatestNodeVersion();
if (engineRange) { if (engineRange) {
const found = const found =
@@ -49,60 +54,35 @@ export async function getSupportedNodeVersion(
return intersects(o.range, engineRange); return intersects(o.range, engineRange);
}); });
if (!found) { if (!found) {
const intro =
isAuto || !engineRange
? 'This project is using an invalid version of Node.js and must be changed.'
: 'Found `engines` in `package.json` with an invalid Node.js version range: "' +
engineRange +
'".';
throw new NowBuildError({ throw new NowBuildError({
code: 'BUILD_UTILS_NODE_VERSION_INVALID', code: 'BUILD_UTILS_NODE_VERSION_INVALID',
link: link: 'http://vercel.link/node-version',
'https://vercel.com/docs/runtimes#official-runtimes/node-js/node-js-version', message: `Found invalid Node.js Version: "${engineRange}". ${getHint(
message: intro + '\n' + pleaseSet, isAuto
)}`,
}); });
} }
} }
if (isDiscontinued(selection)) { if (isDiscontinued(selection)) {
const intro = const intro = `Node.js Version "${selection.range}" is discontinued and must be upgraded.`;
isAuto || !engineRange
? 'This project is using a discontinued version of Node.js (' +
selection.range +
') and must be upgraded.'
: 'Found `engines` in `package.json` with a discontinued Node.js version range: "' +
engineRange +
'".';
throw new NowBuildError({ throw new NowBuildError({
code: 'BUILD_UTILS_NODE_VERSION_DISCONTINUED', code: 'BUILD_UTILS_NODE_VERSION_DISCONTINUED',
link: link: 'http://vercel.link/node-version',
'https://vercel.com/docs/runtimes#official-runtimes/node-js/node-js-version', message: `${intro} ${getHint(isAuto)} ${upstreamProvider}`,
message: intro + '\n' + pleaseSet + '\n' + upstreamProvider,
}); });
} }
debug( debug(`Selected Node.js ${selection.range}`);
isAuto || !engineRange
? 'Using default Node.js range: "' + selection.range + '".'
: 'Found `engines` in `package.json`, selecting range: "' +
selection.range +
'".'
);
if (selection.discontinueDate) { if (selection.discontinueDate) {
const d = selection.discontinueDate.toISOString().split('T')[0]; const d = selection.discontinueDate.toISOString().split('T')[0];
console.warn( console.warn(
boxen( `Error: Node.js version ${
'NOTICE' + selection.range
'\n' + } is deprecated. Deployments created on or after ${d} will fail to build. ${getHint(
`\nNode.js version ${selection.range} has reached end-of-life.` + isAuto
`\nAs a result, deployments created on or after ${d} will fail to build.` + )} ${upstreamProvider}`
'\n' +
pleaseSet +
'\n' +
upstreamProvider,
{ padding: 1 }
)
); );
} }

View File

@@ -5,7 +5,6 @@ import debug from '../debug';
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 { cpus } from 'os';
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';
@@ -165,8 +164,8 @@ export function getSpawnOptions(
export async function getNodeVersion( export async function getNodeVersion(
destPath: string, destPath: string,
_nodeVersion?: string, _nodeVersion?: string,
_config?: Config, config: Config = {},
meta?: Meta meta: Meta = {}
): Promise<NodeVersion> { ): Promise<NodeVersion> {
if (meta && meta.isDev) { if (meta && meta.isDev) {
// Use the system-installed version of `node` in PATH for `vercel dev` // Use the system-installed version of `node` in PATH for `vercel dev`
@@ -174,13 +173,19 @@ export async function getNodeVersion(
return { ...latest, runtime: 'nodejs' }; return { ...latest, runtime: 'nodejs' };
} }
const { packageJson } = await scanParentDirs(destPath, true); const { packageJson } = await scanParentDirs(destPath, true);
let range: string | undefined; let { nodeVersion } = config;
let isAuto = true; let isAuto = true;
if (packageJson && packageJson.engines && packageJson.engines.node) { if (packageJson && packageJson.engines && packageJson.engines.node) {
range = packageJson.engines.node; const { node } = packageJson.engines;
if (nodeVersion && nodeVersion !== node && !meta.isDev) {
console.warn(
`Warning: Due to "engines": { "node": "${node}" } in your \`package.json\` file, the Node.js Version defined in your Project Settings ("${nodeVersion}") will not apply. Learn More: http://vercel.link/node-version`
);
}
nodeVersion = node;
isAuto = false; isAuto = false;
} }
return getSupportedNodeVersion(range, isAuto); return getSupportedNodeVersion(nodeVersion, isAuto);
} }
async function scanParentDirs(destPath: string, readPackageJson = false) { async function scanParentDirs(destPath: string, readPackageJson = false) {
@@ -319,18 +324,7 @@ export async function runBundleInstall(
assert(path.isAbsolute(destPath)); assert(path.isAbsolute(destPath));
const opts = { ...spawnOpts, cwd: destPath, prettyCommand: 'bundle install' }; const opts = { ...spawnOpts, cwd: destPath, prettyCommand: 'bundle install' };
await spawnAsync( await spawnAsync('bundle', args.concat(['install']), opts);
'bundle',
args.concat([
'install',
'--no-prune',
'--retry',
'3',
'--jobs',
String(cpus().length || 1),
]),
opts
);
} }
export async function runPipInstall( export async function runPipInstall(
@@ -394,7 +388,7 @@ export async function runPackageJsonScript(
}); });
} else { } else {
// Yarn v2 PnP mode may be activated, so force "node-modules" linker style // Yarn v2 PnP mode may be activated, so force "node-modules" linker style
const env: typeof process.env = { ...spawnOpts?.env }; const env: typeof process.env = { ...process.env, ...spawnOpts?.env };
if (!env.YARN_NODE_LINKER) { if (!env.YARN_NODE_LINKER) {
env.YARN_NODE_LINKER = 'node-modules'; env.YARN_NODE_LINKER = 'node-modules';
} }

View File

@@ -1,3 +1,4 @@
/env.txt
/node_modules/ /node_modules/
/public/build/ /public/build/

View File

@@ -4,7 +4,8 @@
"scripts": { "scripts": {
"build": "rollup -c", "build": "rollup -c",
"dev": "rollup -c -w", "dev": "rollup -c -w",
"start": "sirv public" "start": "sirv public",
"env": "echo \"$PATH\" > env.txt"
}, },
"devDependencies": { "devDependencies": {
"@rollup/plugin-commonjs": "^12.0.0", "@rollup/plugin-commonjs": "^12.0.0",

View File

@@ -39,7 +39,7 @@ for (const fixture of fs.readdirSync(fixturesPath)) {
// few foreign tests // few foreign tests
const buildersToTestWith = ['now-next', 'now-node', 'now-static-build']; const buildersToTestWith = ['now-node'];
// eslint-disable-next-line no-restricted-syntax // eslint-disable-next-line no-restricted-syntax
for (const builder of buildersToTestWith) { for (const builder of buildersToTestWith) {

View File

@@ -0,0 +1,6 @@
{
"private": true,
"engines": {
"node": "14.x"
}
}

View File

@@ -0,0 +1 @@
contents

View File

@@ -0,0 +1 @@
dir

View File

@@ -2572,6 +2572,101 @@ it('Test `detectRoutes` with `featHandleMiss=true`', async () => {
]); ]);
} }
{
const files = ['api/external.js', 'pages/api/internal.js'];
const { builders, warnings } = await detectBuilders(files, null, {
featHandleMiss,
projectSettings: { framework: 'nextjs' },
});
expect(builders).toStrictEqual([
{
config: {
zeroConfig: true,
},
src: 'api/external.js',
use: '@vercel/node',
},
{
config: {
framework: 'nextjs',
zeroConfig: true,
},
src: 'package.json',
use: '@vercel/next',
},
]);
expect(warnings).toStrictEqual([
{
code: 'conflicting_files',
message:
'When using Next.js, it is recommended to place Node.js Serverless Functions inside of the `pages/api` (provided by Next.js) directory instead of `api` (provided by Vercel).',
link: 'https://nextjs.org/docs/api-routes/introduction',
action: 'Learn More',
},
]);
}
{
const files = ['api/external.js', 'pages/api/internal.js'];
const { builders, warnings } = await detectBuilders(files, null, {
featHandleMiss,
tag: 'canary',
projectSettings: { framework: 'nextjs' },
});
expect(builders).toStrictEqual([
{
config: {
zeroConfig: true,
},
src: 'api/external.js',
use: '@vercel/node@canary',
},
{
config: {
framework: 'nextjs',
zeroConfig: true,
},
src: 'package.json',
use: '@vercel/next@canary',
},
]);
expect(warnings).toStrictEqual([
{
code: 'conflicting_files',
message:
'When using Next.js, it is recommended to place Node.js Serverless Functions inside of the `pages/api` (provided by Next.js) directory instead of `api` (provided by Vercel).',
link: 'https://nextjs.org/docs/api-routes/introduction',
action: 'Learn More',
},
]);
}
{
const files = ['api/external.go', 'pages/api/internal.js'];
const { builders, warnings } = await detectBuilders(files, null, {
featHandleMiss,
projectSettings: { framework: 'nextjs' },
});
expect(builders).toStrictEqual([
{
config: {
zeroConfig: true,
},
src: 'api/external.go',
use: '@vercel/go',
},
{
config: {
framework: 'nextjs',
zeroConfig: true,
},
src: 'package.json',
use: '@vercel/next',
},
]);
expect(warnings).toStrictEqual([]);
}
{ {
const files = ['public/index.html']; const files = ['public/index.html'];

View File

@@ -1,14 +1,6 @@
import path from 'path'; import frameworkList from '@vercel/frameworks';
import { readFileSync } from 'fs-extra';
import { Framework } from '@vercel/frameworks';
import { detectFramework, DetectorFilesystem } from '../src'; import { detectFramework, DetectorFilesystem } from '../src';
const frameworkList = JSON.parse(
readFileSync(
path.join(__dirname, '..', '..', 'frameworks', 'frameworks.json')
).toString()
) as Framework[];
class VirtualFilesystem extends DetectorFilesystem { class VirtualFilesystem extends DetectorFilesystem {
private files: Map<string, Buffer>; private files: Map<string, Buffer>;
@@ -98,7 +90,7 @@ describe('#detectFramework', () => {
it('Detect Hugo #1', async () => { it('Detect Hugo #1', async () => {
const fs = new VirtualFilesystem({ const fs = new VirtualFilesystem({
'config.yaml': 'config', 'config.yaml': 'baseURL: http://example.org/',
'content/post.md': '# hello world', 'content/post.md': '# hello world',
}); });
@@ -107,7 +99,7 @@ describe('#detectFramework', () => {
it('Detect Hugo #2', async () => { it('Detect Hugo #2', async () => {
const fs = new VirtualFilesystem({ const fs = new VirtualFilesystem({
'config.json': 'config', 'config.json': '{ "baseURL": "http://example.org/" }',
'content/post.md': '# hello world', 'content/post.md': '# hello world',
}); });
@@ -116,7 +108,7 @@ describe('#detectFramework', () => {
it('Detect Hugo #3', async () => { it('Detect Hugo #3', async () => {
const fs = new VirtualFilesystem({ const fs = new VirtualFilesystem({
'config.toml': 'config', 'config.toml': 'baseURL = "http://example.org/"',
'content/post.md': '# hello world', 'content/post.md': '# hello world',
}); });
@@ -151,4 +143,12 @@ describe('#detectFramework', () => {
expect(await detectFramework({ fs, frameworkList })).toBe('scully'); expect(await detectFramework({ fs, frameworkList })).toBe('scully');
}); });
it('Detect Zola', async () => {
const fs = new VirtualFilesystem({
'config.toml': 'base_url = "/"',
});
expect(await detectFramework({ fs, frameworkList })).toBe('zola');
});
}); });

View File

@@ -1,3 +1,4 @@
const ms = require('ms');
const path = require('path'); const path = require('path');
const fs = require('fs-extra'); const fs = require('fs-extra');
const assert = require('assert').strict; const assert = require('assert').strict;
@@ -8,6 +9,8 @@ const {
getNodeVersion, getNodeVersion,
getLatestNodeVersion, getLatestNodeVersion,
getDiscontinuedNodeVersions, getDiscontinuedNodeVersions,
runNpmInstall,
runPackageJsonScript,
} = require('../dist'); } = require('../dist');
async function expectBuilderError(promise, pattern) { async function expectBuilderError(promise, pattern) {
@@ -19,30 +22,49 @@ async function expectBuilderError(promise, pattern) {
} }
assert('message' in result, `Expected error message but found ${result}`); assert('message' in result, `Expected error message but found ${result}`);
assert( assert(
pattern.test(result.message), typeof result.message === 'string',
`Expected error to be a string but found ${typeof result.message}`
);
assert(
result.message.includes(pattern),
`Expected ${pattern} but found "${result.message}"` `Expected ${pattern} but found "${result.message}"`
); );
} }
let warningMessages;
const originalConsoleWarn = console.warn;
beforeEach(() => {
warningMessages = [];
console.warn = m => {
warningMessages.push(m);
};
});
afterEach(() => {
console.warn = originalConsoleWarn;
});
it('should re-create symlinks properly', async () => { it('should re-create 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;
} }
const files = await glob('**', path.join(__dirname, 'symlinks')); const files = await glob('**', path.join(__dirname, 'symlinks'));
assert.equal(Object.keys(files).length, 2); assert.equal(Object.keys(files).length, 4);
const outDir = path.join(__dirname, 'symlinks-out'); const outDir = path.join(__dirname, 'symlinks-out');
await fs.remove(outDir); await fs.remove(outDir);
const files2 = await download(files, outDir); const files2 = await download(files, outDir);
assert.equal(Object.keys(files2).length, 2); assert.equal(Object.keys(files2).length, 4);
const [linkStat, aStat] = await Promise.all([ const [linkStat, linkDirStat, aStat] = await Promise.all([
fs.lstat(path.join(outDir, 'link.txt')), 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, 'a.txt')),
]); ]);
assert(linkStat.isSymbolicLink()); assert(linkStat.isSymbolicLink());
assert(linkDirStat.isSymbolicLink());
assert(aStat.isFile()); assert(aStat.isFile());
}); });
@@ -52,7 +74,7 @@ it('should create zip files with symlinks properly', async () => {
return; return;
} }
const files = await glob('**', path.join(__dirname, 'symlinks')); const files = await glob('**', path.join(__dirname, 'symlinks'));
assert.equal(Object.keys(files).length, 2); assert.equal(Object.keys(files).length, 4);
const outFile = path.join(__dirname, 'symlinks.zip'); const outFile = path.join(__dirname, 'symlinks.zip');
await fs.remove(outFile); await fs.remove(outFile);
@@ -64,15 +86,17 @@ it('should create zip files with symlinks properly', async () => {
await fs.writeFile(outFile, await createZip(files)); await fs.writeFile(outFile, await createZip(files));
await spawnAsync('unzip', [outFile], { cwd: outDir }); await spawnAsync('unzip', [outFile], { cwd: outDir });
const [linkStat, aStat] = await Promise.all([ const [linkStat, linkDirStat, aStat] = await Promise.all([
fs.lstat(path.join(outDir, 'link.txt')), 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, 'a.txt')),
]); ]);
assert(linkStat.isSymbolicLink()); assert(linkStat.isSymbolicLink());
assert(linkDirStat.isSymbolicLink());
assert(aStat.isFile()); assert(aStat.isFile());
}); });
it('should only match supported node versions', async () => { it('should only match supported node versions, otherwise throw an error', async () => {
expect(await getSupportedNodeVersion('10.x', false)).toHaveProperty( expect(await getSupportedNodeVersion('10.x', false)).toHaveProperty(
'major', 'major',
10 10
@@ -81,12 +105,13 @@ it('should only match supported node versions', async () => {
'major', 'major',
12 12
); );
expect(getSupportedNodeVersion('8.11.x', false)).rejects.toThrow(); expect(await getSupportedNodeVersion('14.x', false)).toHaveProperty(
expect(getSupportedNodeVersion('6.x', false)).rejects.toThrow(); 'major',
expect(getSupportedNodeVersion('999.x', false)).rejects.toThrow(); 14
expect(getSupportedNodeVersion('foo', false)).rejects.toThrow(); );
const autoMessage = /This project is using an invalid version of Node.js and must be changed/; const autoMessage =
'Please set Node.js Version to 14.x in your Project Settings to use Node.js 14.';
await expectBuilderError( await expectBuilderError(
getSupportedNodeVersion('8.11.x', true), getSupportedNodeVersion('8.11.x', true),
autoMessage autoMessage
@@ -104,7 +129,13 @@ it('should only match supported node versions', async () => {
'major', 'major',
12 12
); );
const foundMessage = /Found `engines` in `package\.json` with an invalid Node\.js version range/; expect(await getSupportedNodeVersion('14.x', true)).toHaveProperty(
'major',
14
);
const foundMessage =
'Please set "engines": { "node": "14.x" } in your `package.json` file to use Node.js 14.';
await expectBuilderError( await expectBuilderError(
getSupportedNodeVersion('8.11.x', false), getSupportedNodeVersion('8.11.x', false),
foundMessage foundMessage
@@ -125,8 +156,8 @@ it('should match all semver ranges', async () => {
// See https://docs.npmjs.com/files/package.json#engines // See https://docs.npmjs.com/files/package.json#engines
expect(await getSupportedNodeVersion('10.0.0')).toHaveProperty('major', 10); expect(await getSupportedNodeVersion('10.0.0')).toHaveProperty('major', 10);
expect(await getSupportedNodeVersion('10.x')).toHaveProperty('major', 10); expect(await getSupportedNodeVersion('10.x')).toHaveProperty('major', 10);
expect(await getSupportedNodeVersion('>=10')).toHaveProperty('major', 12); expect(await getSupportedNodeVersion('>=10')).toHaveProperty('major', 14);
expect(await getSupportedNodeVersion('>=10.3.0')).toHaveProperty('major', 12); expect(await getSupportedNodeVersion('>=10.3.0')).toHaveProperty('major', 14);
expect(await getSupportedNodeVersion('8.5.0 - 10.5.0')).toHaveProperty( expect(await getSupportedNodeVersion('8.5.0 - 10.5.0')).toHaveProperty(
'major', 'major',
10 10
@@ -150,20 +181,78 @@ it('should ignore node version in vercel dev getNodeVersion()', async () => {
).toHaveProperty('runtime', 'nodejs'); ).toHaveProperty('runtime', 'nodejs');
}); });
it('should select project setting from config when no package.json is found', async () => {
expect(
await getNodeVersion('/tmp', undefined, { nodeVersion: '14.x' }, {})
).toHaveProperty('range', '14.x');
expect(warningMessages).toStrictEqual([]);
});
it('should prefer package.json engines over project setting from config and warn', async () => {
expect(
await getNodeVersion(
path.join(__dirname, 'pkg-engine-node'),
undefined,
{ nodeVersion: '12.x' },
{}
)
).toHaveProperty('range', '14.x');
expect(warningMessages).toStrictEqual([
'Warning: Due to "engines": { "node": "14.x" } in your `package.json` file, the Node.js Version defined in your Project Settings ("12.x") will not apply. Learn More: http://vercel.link/node-version',
]);
});
it('should not warn when package.json engines matches project setting from config', async () => {
expect(
await getNodeVersion(
path.join(__dirname, 'pkg-engine-node'),
undefined,
{ nodeVersion: '14.x' },
{}
)
).toHaveProperty('range', '14.x');
expect(warningMessages).toStrictEqual([]);
});
it('should get latest node version', async () => { it('should get latest node version', async () => {
expect(await getLatestNodeVersion()).toHaveProperty('major', 12); expect(await getLatestNodeVersion()).toHaveProperty('major', 14);
}); });
it('should throw for discontinued versions', async () => { it('should throw for discontinued versions', async () => {
// Mock a future date so that Node 8 becomes discontinued // Mock a future date so that Node 8 and 10 become discontinued
const realDateNow = Date.now.bind(global.Date); const realDateNow = Date.now.bind(global.Date);
global.Date.now = () => new Date('2020-02-14').getTime(); global.Date.now = () => new Date('2021-05-01').getTime();
expect(getSupportedNodeVersion('8.10.x', false)).rejects.toThrow(); expect(getSupportedNodeVersion('8.10.x', false)).rejects.toThrow();
expect(getSupportedNodeVersion('8.10.x', true)).rejects.toThrow(); expect(getSupportedNodeVersion('8.10.x', true)).rejects.toThrow();
expect(getSupportedNodeVersion('10.x', false)).rejects.toThrow();
expect(getSupportedNodeVersion('10.x', true)).rejects.toThrow();
expect(getDiscontinuedNodeVersions().length).toBe(1); const discontinued = getDiscontinuedNodeVersions();
expect(getDiscontinuedNodeVersions()[0]).toHaveProperty('range', '8.10.x'); expect(discontinued.length).toBe(2);
expect(discontinued[0]).toHaveProperty('range', '10.x');
expect(discontinued[1]).toHaveProperty('range', '8.10.x');
global.Date.now = realDateNow;
});
it('should warn for deprecated versions, soon to be discontinued', async () => {
// Mock a future date so that Node 10 warns
const realDateNow = Date.now.bind(global.Date);
global.Date.now = () => new Date('2021-02-23').getTime();
expect(await getSupportedNodeVersion('10.x', false)).toHaveProperty(
'major',
10
);
expect(await getSupportedNodeVersion('10.x', true)).toHaveProperty(
'major',
10
);
expect(warningMessages).toStrictEqual([
'Error: Node.js version 10.x is deprecated. Deployments created on or after 2021-04-20 will fail to build. Please set "engines": { "node": "14.x" } in your `package.json` file to use Node.js 14. This change is the result of a decision made by an upstream infrastructure provider (AWS).',
'Error: Node.js version 10.x is deprecated. Deployments created on or after 2021-04-20 will fail to build. Please set Node.js Version to 14.x in your Project Settings to use Node.js 14. This change is the result of a decision made by an upstream infrastructure provider (AWS).',
]);
global.Date.now = realDateNow; global.Date.now = realDateNow;
}); });
@@ -197,3 +286,18 @@ it('should support require by path for legacy builders', () => {
expect(FileRef2).toBe(index.FileRef); expect(FileRef2).toBe(index.FileRef);
expect(Lambda2).toBe(index.Lambda); expect(Lambda2).toBe(index.Lambda);
}); });
it(
'should have correct $PATH when running `runPackageJsonScript()` with yarn',
async () => {
const fixture = path.join(__dirname, 'fixtures', '19-yarn-v2');
await runNpmInstall(fixture);
await runPackageJsonScript(fixture, 'env');
// `yarn` was failing with ENOENT before, so as long as the
// script was invoked at all is enough to verify the fix
const out = await fs.readFile(path.join(fixture, 'env.txt'), 'utf8');
expect(out.trim()).toBeTruthy();
},
ms('1m')
);

25
packages/now-cgi/build.js Normal file
View File

@@ -0,0 +1,25 @@
#!/usr/bin/env node
const execa = require('execa');
const { join } = require('path');
const { homedir } = require('os');
async function main() {
process.env.GOOS = 'linux';
process.env.GOARCH = 'amd64';
process.env.GOPATH = join(homedir(), 'go');
await execa('go', ['get', 'github.com/aws/aws-lambda-go/events'], {
stdio: 'inherit',
});
await execa('go', ['get', 'github.com/aws/aws-lambda-go/lambda'], {
stdio: 'inherit',
});
await execa('go', ['build', '-o', 'handler', 'main.go'], {
stdio: 'inherit',
});
}
main().catch(err => {
console.error(err);
process.exit(1);
});

View File

@@ -1,7 +0,0 @@
#!/usr/bin/env bash
export GOOS=linux
export GOARCH=amd64
export GOPATH=$HOME/go
go get github.com/aws/aws-lambda-go/events
go get github.com/aws/aws-lambda-go/lambda
go build -o handler main.go

View File

@@ -1,6 +1,6 @@
{ {
"name": "@vercel/cgi", "name": "@vercel/cgi",
"version": "1.0.6", "version": "1.0.7-canary.0",
"license": "MIT", "license": "MIT",
"repository": { "repository": {
"type": "git", "type": "git",
@@ -8,7 +8,8 @@
"directory": "packages/now-cgi" "directory": "packages/now-cgi"
}, },
"scripts": { "scripts": {
"prepublish": "./build.sh" "build": "node build",
"prepublishOnly": "node build"
}, },
"files": [ "files": [
"index.js", "index.js",

View File

@@ -1,6 +1,6 @@
{ {
"name": "vercel", "name": "vercel",
"version": "21.1.0", "version": "21.3.2",
"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",
@@ -61,11 +61,11 @@
"node": ">= 10" "node": ">= 10"
}, },
"dependencies": { "dependencies": {
"@vercel/build-utils": "2.7.0", "@vercel/build-utils": "2.10.0",
"@vercel/go": "1.1.7", "@vercel/go": "1.2.1",
"@vercel/node": "1.9.0", "@vercel/node": "1.9.1-canary.0",
"@vercel/python": "1.2.4", "@vercel/python": "2.0.0",
"@vercel/ruby": "1.2.5", "@vercel/ruby": "1.2.6-canary.0",
"update-notifier": "4.1.0" "update-notifier": "4.1.0"
}, },
"devDependencies": { "devDependencies": {
@@ -100,7 +100,7 @@
"@types/universal-analytics": "0.4.2", "@types/universal-analytics": "0.4.2",
"@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/frameworks": "0.2.0", "@vercel/frameworks": "0.3.0",
"@vercel/ncc": "0.24.0", "@vercel/ncc": "0.24.0",
"@zeit/fun": "0.11.2", "@zeit/fun": "0.11.2",
"@zeit/source-map-support": "0.6.2", "@zeit/source-map-support": "0.6.2",

View File

@@ -2,7 +2,6 @@ import chalk from 'chalk';
import { handleError } from '../../util/error'; import { handleError } from '../../util/error';
import createOutput from '../../util/output';
import getArgs from '../../util/get-args'; import getArgs from '../../util/get-args';
import getSubcommand from '../../util/get-subcommand'; import getSubcommand from '../../util/get-subcommand';
import logo from '../../util/output/logo'; import logo from '../../util/output/logo';
@@ -92,15 +91,14 @@ export default async function main(ctx) {
return 2; return 2;
} }
const output = createOutput({ debug: argv['--debug'] });
const { subcommand, args } = getSubcommand(argv._.slice(1), COMMAND_CONFIG); const { subcommand, args } = getSubcommand(argv._.slice(1), COMMAND_CONFIG);
switch (subcommand) { switch (subcommand) {
case 'ls': case 'ls':
return ls(ctx, argv, args, output); return ls(ctx, argv, args);
case 'rm': case 'rm':
return rm(ctx, argv, args, output); return rm(ctx, argv, args);
default: default:
return set(ctx, argv, args, output); return set(ctx, argv, args);
} }
} }

View File

@@ -10,9 +10,10 @@ import strlen from '../../util/strlen.ts';
import getCommandFlags from '../../util/get-command-flags'; import getCommandFlags from '../../util/get-command-flags';
import { getCommandName } from '../../util/pkg-name.ts'; import { getCommandName } from '../../util/pkg-name.ts';
export default async function ls(ctx, opts, args, output) { export default async function ls(ctx, opts, args) {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
} = ctx; } = ctx;
const { currentTeam } = config; const { currentTeam } = config;
@@ -23,6 +24,7 @@ export default async function ls(ctx, opts, args, output) {
token, token,
currentTeam, currentTeam,
debug: debugEnabled, debug: debugEnabled,
output,
}); });
let contextName = null; let contextName = null;
@@ -47,9 +49,9 @@ export default async function ls(ctx, opts, args, output) {
token, token,
debug: debugEnabled, debug: debugEnabled,
currentTeam, currentTeam,
output,
}); });
const lsStamp = stamp(); const lsStamp = stamp();
let cancelWait;
if (args.length > 0) { if (args.length > 0) {
output.error( output.error(
@@ -60,17 +62,13 @@ export default async function ls(ctx, opts, args, output) {
return 1; return 1;
} }
cancelWait = output.spinner( output.spinner(`Fetching aliases under ${chalk.bold(contextName)}`);
`Fetching aliases under ${chalk.bold(contextName)}`
);
const { aliases, pagination } = await getAliases( const { aliases, pagination } = await getAliases(
now, now,
undefined, undefined,
nextTimestamp nextTimestamp
); );
if (cancelWait) cancelWait();
output.log(`aliases found under ${chalk.bold(contextName)} ${lsStamp()}`); output.log(`aliases found under ${chalk.bold(contextName)} ${lsStamp()}`);
console.log(printAliasTable(aliases)); console.log(printAliasTable(aliases));

View File

@@ -12,9 +12,10 @@ import { isValidName } from '../../util/is-valid-name';
import findAliasByAliasOrId from '../../util/alias/find-alias-by-alias-or-id'; import findAliasByAliasOrId from '../../util/alias/find-alias-by-alias-or-id';
import { getCommandName } from '../../util/pkg-name.ts'; import { getCommandName } from '../../util/pkg-name.ts';
export default async function rm(ctx, opts, args, output) { export default async function rm(ctx, opts, args) {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
} = ctx; } = ctx;
const { currentTeam } = config; const { currentTeam } = config;
@@ -25,6 +26,7 @@ export default async function rm(ctx, opts, args, output) {
token, token,
currentTeam, currentTeam,
debug: debugEnabled, debug: debugEnabled,
output,
}); });
let contextName = null; let contextName = null;
@@ -39,8 +41,13 @@ export default async function rm(ctx, opts, args, output) {
throw err; throw err;
} }
// $FlowFixMe const now = new Now({
const now = new Now({ apiUrl, token, debug: debugEnabled, currentTeam }); apiUrl,
token,
debug: debugEnabled,
currentTeam,
output,
});
const [aliasOrId] = args; const [aliasOrId] = args;
if (args.length !== 1) { if (args.length !== 1) {

View File

@@ -29,11 +29,11 @@ type Options = {
export default async function set( export default async function set(
ctx: NowContext, ctx: NowContext,
opts: Options, opts: Options,
args: string[], args: string[]
output: Output
) { ) {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
localConfig, localConfig,
} = ctx; } = ctx;
@@ -49,6 +49,7 @@ export default async function set(
token, token,
currentTeam, currentTeam,
debug: debugEnabled, debug: debugEnabled,
output,
}); });
let user: User; let user: User;

View File

@@ -14,7 +14,6 @@ import addBilling from './add';
import exit from '../../util/exit'; import exit from '../../util/exit';
import Client from '../../util/client.ts'; import Client from '../../util/client.ts';
import getScope from '../../util/get-scope.ts'; import getScope from '../../util/get-scope.ts';
import createOutput from '../../util/output';
import { getPkgName } from '../../util/pkg-name.ts'; import { getPkgName } from '../../util/pkg-name.ts';
const help = () => { const help = () => {
@@ -90,8 +89,9 @@ function buildInquirerChoices(cards) {
const _default = const _default =
source.id === cards.defaultSource ? ` ${chalk.bold('(default)')}` : ''; source.id === cards.defaultSource ? ` ${chalk.bold('(default)')}` : '';
const id = `${chalk.cyan(`ID: ${source.id}`)}${_default}`; const id = `${chalk.cyan(`ID: ${source.id}`)}${_default}`;
const number = `${chalk.gray('#### ').repeat(3)}${source.last4 || const number = `${chalk.gray('#### ').repeat(3)}${
source.card.last4}`; source.last4 || source.card.last4
}`;
const str = [ const str = [
id, id,
indent(source.name || source.owner.name, 2), indent(source.name || source.owner.name, 2),
@@ -106,11 +106,16 @@ function buildInquirerChoices(cards) {
}); });
} }
async function run({ token, config: { currentTeam } }) { async function run({ token, output, config: { currentTeam } }) {
const start = new Date(); const start = new Date();
const creditCards = new NowCreditCards({ apiUrl, token, debug, currentTeam }); const creditCards = new NowCreditCards({
const output = createOutput({ debug }); apiUrl,
const client = new Client({ apiUrl, token, currentTeam, debug }); token,
debug,
currentTeam,
output,
});
const client = new Client({ apiUrl, token, currentTeam, debug, output });
let contextName = null; let contextName = null;
try { try {
@@ -147,8 +152,9 @@ async function run({ token, config: { currentTeam } }) {
const id = `${chalk.gray('-')} ${chalk.cyan( const id = `${chalk.gray('-')} ${chalk.cyan(
`ID: ${source.id}` `ID: ${source.id}`
)}${_default}`; )}${_default}`;
const number = `${chalk.gray('#### ').repeat(3)}${source.last4 || const number = `${chalk.gray('#### ').repeat(3)}${
source.card.last4}`; source.last4 || source.card.last4
}`;
return [ return [
id, id,
@@ -231,8 +237,9 @@ async function run({ token, config: { currentTeam } }) {
const elapsed = ms(new Date() - start); const elapsed = ms(new Date() - start);
console.log( console.log(
success( success(
`${card.brand || card.card.brand} ending in ${card.last4 || `${card.brand || card.card.brand} ending in ${
card.card.last4} is now the default ${chalk.gray(`[${elapsed}]`)}` card.last4 || card.card.last4
} is now the default ${chalk.gray(`[${elapsed}]`)}`
) )
); );
} else { } else {
@@ -301,9 +308,9 @@ async function run({ token, config: { currentTeam } }) {
const deletedCard = cards.sources.find(card => card.id === cardId); const deletedCard = cards.sources.find(card => card.id === cardId);
const remainingCards = cards.sources.filter(card => card.id !== cardId); const remainingCards = cards.sources.filter(card => card.id !== cardId);
let text = `${deletedCard.brand || let text = `${deletedCard.brand || deletedCard.card.brand} ending in ${
deletedCard.card.brand} ending in ${deletedCard.last4 || deletedCard.last4 || deletedCard.card.last4
deletedCard.card.last4} was deleted`; } was deleted`;
// ${chalk.gray(`[${elapsed}]`)} // ${chalk.gray(`[${elapsed}]`)}
if (cardId === cards.defaultSource) { if (cardId === cards.defaultSource) {
@@ -317,11 +324,11 @@ async function run({ token, config: { currentTeam } }) {
card => card.id === cards.defaultCardId card => card.id === cards.defaultCardId
); );
text += `\n${newDefaultCard.brand || text += `\n${
newDefaultCard.card.brand} ending in ${newDefaultCard.last4 || newDefaultCard.brand || newDefaultCard.card.brand
newDefaultCard.card.last4} in now default for ${chalk.bold( } ending in ${
contextName newDefaultCard.last4 || newDefaultCard.card.last4
)}`; } in now default for ${chalk.bold(contextName)}`;
} }
} }

View File

@@ -8,7 +8,6 @@ import stamp from '../../util/output/stamp';
import createCertFromFile from '../../util/certs/create-cert-from-file'; import createCertFromFile from '../../util/certs/create-cert-from-file';
import createCertForCns from '../../util/certs/create-cert-for-cns'; import createCertForCns from '../../util/certs/create-cert-for-cns';
import { NowContext } from '../../types'; import { NowContext } from '../../types';
import { Output } from '../../util/output';
import { getCommandName } from '../../util/pkg-name'; import { getCommandName } from '../../util/pkg-name';
interface Options { interface Options {
@@ -22,11 +21,11 @@ interface Options {
async function add( async function add(
ctx: NowContext, ctx: NowContext,
opts: Options, opts: Options,
args: string[], args: string[]
output: Output
): Promise<number> { ): Promise<number> {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
} = ctx; } = ctx;
const { currentTeam } = config; const { currentTeam } = config;
@@ -49,6 +48,7 @@ async function add(
token, token,
currentTeam, currentTeam,
debug: debugEnabled, debug: debugEnabled,
output,
}); });
try { try {
@@ -62,7 +62,13 @@ async function add(
throw err; throw err;
} }
const now = new Now({ apiUrl, token, debug: debugEnabled, currentTeam }); const now = new Now({
apiUrl,
token,
debug: debugEnabled,
currentTeam,
output,
});
if (overwite) { if (overwite) {
output.error('Overwrite option is deprecated'); output.error('Overwrite option is deprecated');
@@ -71,7 +77,7 @@ async function add(
} }
if (crtPath || keyPath || caPath) { if (crtPath || keyPath || caPath) {
if (args.length !== 0 || (!crtPath || !keyPath || !caPath)) { if (args.length !== 0 || !crtPath || !keyPath || !caPath) {
output.error( output.error(
`Invalid number of arguments to create a custom certificate entry. Usage:` `Invalid number of arguments to create a custom certificate entry. Usage:`
); );
@@ -113,12 +119,12 @@ async function add(
(res, item) => res.concat(item.split(',')), (res, item) => res.concat(item.split(',')),
[] []
); );
const cancelWait = output.spinner( output.spinner(
`Generating a certificate for ${chalk.bold(cns.join(', '))}` `Generating a certificate for ${chalk.bold(cns.join(', '))}`
); );
cert = await createCertForCns(now, cns, contextName); cert = await createCertForCns(now, cns, contextName);
cancelWait(); output.stopSpinner();
} }
if (cert instanceof Error) { if (cert instanceof Error) {

View File

@@ -3,7 +3,6 @@ import chalk from 'chalk';
// @ts-ignore // @ts-ignore
import { handleError } from '../../util/error'; import { handleError } from '../../util/error';
import createOutput from '../../util/output';
import getArgs from '../../util/get-args'; import getArgs from '../../util/get-args';
import getSubcommand from '../../util/get-subcommand'; import getSubcommand from '../../util/get-subcommand';
import logo from '../../util/output/logo'; import logo from '../../util/output/logo';
@@ -104,17 +103,17 @@ export default async function main(ctx: NowContext) {
return 0; return 0;
} }
const output = createOutput({ debug: argv['--debug'] }); const { output } = ctx;
const { subcommand, args } = getSubcommand(argv._.slice(1), COMMAND_CONFIG); const { subcommand, args } = getSubcommand(argv._.slice(1), COMMAND_CONFIG);
switch (subcommand) { switch (subcommand) {
case 'issue': case 'issue':
return issue(ctx, argv, args, output); return issue(ctx, argv, args);
case 'ls': case 'ls':
return ls(ctx, argv, args, output); return ls(ctx, argv, args);
case 'rm': case 'rm':
return rm(ctx, argv, args, output); return rm(ctx, argv, args);
case 'add': case 'add':
return add(ctx, argv, args, output); return add(ctx, argv, args);
case 'renew': case 'renew':
output.error('Renewing certificates is deprecated, issue a new one.'); output.error('Renewing certificates is deprecated, issue a new one.');
return 1; return 1;

View File

@@ -28,11 +28,11 @@ type Options = {
export default async function issue( export default async function issue(
ctx: NowContext, ctx: NowContext,
opts: Options, opts: Options,
args: string[], args: string[]
output: Output
) { ) {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
} = ctx; } = ctx;
const { currentTeam } = config; const { currentTeam } = config;
@@ -55,6 +55,7 @@ export default async function issue(
token, token,
currentTeam, currentTeam,
debug: debugEnabled, debug: debugEnabled,
output,
}); });
let contextName = null; let contextName = null;

View File

@@ -8,7 +8,6 @@ import getScope from '../../util/get-scope';
import stamp from '../../util/output/stamp'; import stamp from '../../util/output/stamp';
import getCerts from '../../util/certs/get-certs'; import getCerts from '../../util/certs/get-certs';
import strlen from '../../util/strlen'; import strlen from '../../util/strlen';
import { Output } from '../../util/output';
import { NowContext, Cert } from '../../types'; import { NowContext, Cert } from '../../types';
import getCommandFlags from '../../util/get-command-flags'; import getCommandFlags from '../../util/get-command-flags';
import { getCommandName } from '../../util/pkg-name'; import { getCommandName } from '../../util/pkg-name';
@@ -21,17 +20,17 @@ interface Options {
async function ls( async function ls(
ctx: NowContext, ctx: NowContext,
opts: Options, opts: Options,
args: string[], args: string[]
output: Output
): Promise<number> { ): Promise<number> {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
} = ctx; } = ctx;
const { currentTeam } = config; const { currentTeam } = config;
const { apiUrl } = ctx; const { apiUrl } = ctx;
const { '--debug': debug, '--next': nextTimestamp } = opts; const { '--debug': debug, '--next': nextTimestamp } = opts;
const client = new Client({ apiUrl, token, currentTeam, debug }); const client = new Client({ apiUrl, token, currentTeam, debug, output });
let contextName = null; let contextName = null;
try { try {
@@ -48,7 +47,7 @@ async function ls(
output.error('Please provide a number for flag --next'); output.error('Please provide a number for flag --next');
return 1; return 1;
} }
const now = new Now({ apiUrl, token, debug, currentTeam }); const now = new Now({ apiUrl, token, debug, currentTeam, output });
const lsStamp = stamp(); const lsStamp = stamp();
if (args.length !== 0) { if (args.length !== 0) {

View File

@@ -18,21 +18,17 @@ type Options = {
'--debug': boolean; '--debug': boolean;
}; };
async function rm( async function rm(ctx: NowContext, opts: Options, args: string[]) {
ctx: NowContext,
opts: Options,
args: string[],
output: Output
) {
const { const {
authConfig: { token }, authConfig: { token },
output,
config, config,
} = ctx; } = ctx;
const { currentTeam } = config; const { currentTeam } = config;
const { apiUrl } = ctx; const { apiUrl } = ctx;
const rmStamp = stamp(); const rmStamp = stamp();
const debug = opts['--debug']; const debug = opts['--debug'];
const client = new Client({ apiUrl, token, currentTeam, debug }); const client = new Client({ apiUrl, token, currentTeam, debug, output });
let contextName = null; let contextName = null;

View File

@@ -3,7 +3,6 @@ import { resolve, basename } from 'path';
import { fileNameSymbol } from '@vercel/client'; import { fileNameSymbol } from '@vercel/client';
import Client from '../../util/client.ts'; import Client from '../../util/client.ts';
import getScope from '../../util/get-scope.ts'; import getScope from '../../util/get-scope.ts';
import createOutput from '../../util/output';
import code from '../../util/output/code'; import code from '../../util/output/code';
import highlight from '../../util/output/highlight'; import highlight from '../../util/output/highlight';
import { readLocalConfig } from '../../util/config/files'; import { readLocalConfig } from '../../util/config/files';
@@ -15,6 +14,7 @@ import deploy from './latest';
export default async ctx => { export default async ctx => {
const { const {
authConfig, authConfig,
output,
config: { currentTeam }, config: { currentTeam },
apiUrl, apiUrl,
} = ctx; } = ctx;
@@ -48,7 +48,6 @@ export default async ctx => {
localConfig = readLocalConfig(paths[0]); localConfig = readLocalConfig(paths[0]);
} }
const debugEnabled = argv['--debug']; const debugEnabled = argv['--debug'];
const output = createOutput({ debug: debugEnabled });
const stats = {}; const stats = {};
if (argv['--help']) { if (argv['--help']) {
@@ -74,6 +73,7 @@ export default async ctx => {
apiUrl, apiUrl,
token: authConfig.token, token: authConfig.token,
currentTeam, currentTeam,
output,
debug: debugEnabled, debug: debugEnabled,
}); });
try { try {

View File

@@ -233,8 +233,7 @@ export default async function main(
const paths = Object.keys(stats); const paths = Object.keys(stats);
const debugEnabled = argv['--debug']; const debugEnabled = argv['--debug'];
// $FlowFixMe const { isTTY } = process.stdout;
const isTTY = process.stdout.isTTY;
const quiet = !isTTY; const quiet = !isTTY;
// check paths // check paths
@@ -263,6 +262,7 @@ export default async function main(
apiUrl: ctx.apiUrl, apiUrl: ctx.apiUrl,
token: ctx.authConfig.token, token: ctx.authConfig.token,
debug: debugEnabled, debug: debugEnabled,
output,
}); });
// retrieve `project` and `org` from .vercel // retrieve `project` and `org` from .vercel
@@ -645,6 +645,7 @@ export default async function main(
token: ctx.authConfig.token, token: ctx.authConfig.token,
currentTeam: org.id, currentTeam: org.id,
debug: debugEnabled, debug: debugEnabled,
output,
}), }),
err.meta.domain, err.meta.domain,
contextName contextName
@@ -727,6 +728,7 @@ export default async function main(
token: ctx.authConfig.token, token: ctx.authConfig.token,
currentTeam: org.type === 'team' ? org.id : null, currentTeam: org.type === 'team' ? org.id : null,
debug: debugEnabled, debug: debugEnabled,
output,
}), }),
deployment, deployment,
deployStamp, deployStamp,

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