mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-19 22:51:41 +08:00
commit
bc48607e75
@ -1,539 +0,0 @@
|
|||||||
translating----geekpi
|
|
||||||
|
|
||||||
UP – deploy serverless apps in seconds
|
|
||||||
============================================================
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/2000/1*8KijrYCm1j0_XvrACQD_fQ.png)
|
|
||||||
|
|
||||||
Last year I wrote [Blueprints for Up][1], describing how most of the building blocks are available to create a great serverless experience on AWS with minimal effort. This post talks about the initial alpha release of [Up][2].
|
|
||||||
|
|
||||||
Why focus on serverless? For starters it’s cost-effective since you pay on-demand, only for what you use. Serverless options are self-healing, as each request is isolated and considered to be “stateless.” And finally it scales indefinitely with ease — there are no machines or clusters to manage. Deploy your code and you’re done.
|
|
||||||
|
|
||||||
Roughly a month ago I decided to start working on it over at [apex/up][3], and wrote the first small serverless sample application [tj/gh-polls][4] for live SVG GitHub user polls. It worked well and costs less than $1/month to serve millions of polls, so I thought I’d go ahead with the project and see if I can offer open-source and commercial variants.
|
|
||||||
|
|
||||||
The long-term goal is to provide a “Bring your own Heroku” of sorts, supporting many platforms. While Platform-as-a-Service is nothing new, the serverless ecosystem is making this kind of program increasingly trivial. This said, AWS and others often suffer in terms of UX due to the flexibility they provide. Up abstracts the complexity away, while still providing you with a virtually ops-free solution.
|
|
||||||
|
|
||||||
### Installation
|
|
||||||
|
|
||||||
You can install Up with the following command, and view the [temporary documentation][5] to get started. Or if you’re sketched out by install scripts, grab a [binary release][6]. (Keep in mind that this project is still early on.)
|
|
||||||
|
|
||||||
```
|
|
||||||
curl -sfL https://raw.githubusercontent.com/apex/up/master/install.sh | sh
|
|
||||||
```
|
|
||||||
|
|
||||||
To upgrade to the latest version at any time just run:
|
|
||||||
|
|
||||||
```
|
|
||||||
up upgrade
|
|
||||||
```
|
|
||||||
|
|
||||||
You may also install via NPM:
|
|
||||||
|
|
||||||
```
|
|
||||||
npm install -g up
|
|
||||||
```
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
What features does the early alpha provide? Let’s take a look! Keep in mind that Up is not a hosted service, so you’ll need an AWS account and [AWS credentials][8]. If you’re not familiar at all with AWS you may want to hold off until that process is streamlined.
|
|
||||||
|
|
||||||
The first question I always get is: how does up(1) differ from [apex(1)][9]? Apex focuses on deploying functions, for pipelines and event processing, while Up focuses on apps, apis, and static sites, aka single deployable units. Apex does not provision API Gateway, SSL certs, or DNS for you, nor does it provide URL rewriting, script injection and so on.
|
|
||||||
|
|
||||||
#### Single command serverless apps
|
|
||||||
|
|
||||||
Up lets you deploy apps, apis, and static sites with a single command. To create an application all you need is a single file, in the case of Node.js, an `./app.js` listening on `PORT` which is provided by Up. Note that if you’re using a `package.json` Up will detect and utilize the `start` and `build`scripts.
|
|
||||||
|
|
||||||
```
|
|
||||||
const http = require('http')
|
|
||||||
const { PORT = 3000 } = process.env
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
http.createServer((req, res) => {
|
|
||||||
res.end('Hello World\n')
|
|
||||||
}).listen(PORT)
|
|
||||||
```
|
|
||||||
|
|
||||||
Additional [runtimes][10] are supported out of the box, such as `main.go` for Golang, so you can deploy Golang, Python, Crystal, or Node.js applications in seconds.
|
|
||||||
|
|
||||||
```
|
|
||||||
package main
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"log"
|
|
||||||
"net/http"
|
|
||||||
"os"
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
func main() {
|
|
||||||
addr := ":" + os.Getenv("PORT")
|
|
||||||
http.HandleFunc("/", hello)
|
|
||||||
log.Fatal(http.ListenAndServe(addr, nil))
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
func hello(w http.ResponseWriter, r *http.Request) {
|
|
||||||
fmt.Fprintln(w, "Hello World from Go")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
To deploy the application type `up` to create the resources required, and deploy the application itself. There are no smoke and mirrors here, once it says “complete”, you’re done, the app is immediately available — there is no remote build process.
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/2000/1*tBYR5HXeDDVkb_Pv2MCj1A.png)
|
|
||||||
|
|
||||||
The subsequent deploys will be even quicker since the stack is already provisioned:
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/2000/1*2w2WHDTfTT-7GsMtNPklXw.png)
|
|
||||||
|
|
||||||
Test out your app with `up url --open` to view it in the browser, `up url --copy` to save the URL to the clipboard, or try it with curl:
|
|
||||||
|
|
||||||
```
|
|
||||||
curl `up url`
|
|
||||||
Hello World
|
|
||||||
```
|
|
||||||
|
|
||||||
To delete the app and its resources just type `up stack delete`:
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/2000/1*FUdhBTtDHaZ2CEPHR7PGqg.png)
|
|
||||||
|
|
||||||
Deploy to the staging or production environments using `up staging` or `up production` , and `up url --open production` for example. Note that custom domains are not yet available, [they will be shortly][11]. Later you’ll also be able to “promote” a release to other stages.
|
|
||||||
|
|
||||||
#### Reverse proxy
|
|
||||||
|
|
||||||
One feature which makes Up unique is that it doesn’t just simply deploy your code, it places a Golang reverse proxy in front of your application. This provides many features such as URL rewriting, redirection, script injection and more, which we’ll look at further in the post.
|
|
||||||
|
|
||||||
#### Infrastructure as code
|
|
||||||
|
|
||||||
Up follows modern best practices in terms of configuration, as all changes to the infrastructure can be previewed before applying, and the use of IAM policies can also restrict developer access to prevent mishaps. A side benefit is that it helps self-document your infrastructure as well.
|
|
||||||
|
|
||||||
Here’s an example of configuring some (dummy) DNS records and free SSL certificates via AWS ACM which utilizes LetsEncrypt.
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"name": "app",
|
|
||||||
"dns": {
|
|
||||||
"myapp.com": [
|
|
||||||
{
|
|
||||||
"name": "myapp.com",
|
|
||||||
"type": "A",
|
|
||||||
"ttl": 300,
|
|
||||||
"value": ["35.161.83.243"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "blog.myapp.com",
|
|
||||||
"type": "CNAME",
|
|
||||||
"ttl": 300,
|
|
||||||
"value": ["34.209.172.67"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "api.myapp.com",
|
|
||||||
"type": "A",
|
|
||||||
"ttl": 300,
|
|
||||||
"value": ["54.187.185.18"]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"certs": [
|
|
||||||
{
|
|
||||||
"domains": ["myapp.com", "*.myapp.com"]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
When you deploy the application the first time via `up` all the permissions required, API Gateway, Lambda function, ACM certs, Route53 DNS records and others are created for you.
|
|
||||||
|
|
||||||
[ChangeSets][12] are not yet implemented but you will be able to preview further changes with `up stack plan` and commit them with `up stack apply`, much like you would with Terraform.
|
|
||||||
|
|
||||||
Check out the [configuration documentation][13] for more information.
|
|
||||||
|
|
||||||
#### Global deploys
|
|
||||||
|
|
||||||
The `regions` array allows you to specify target regions for your app. For example if you’re only interested in a single region you’d use:
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"regions": ["us-west-2"]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
If your customers are concentrated in North America, you may want to use all of the US and CA regions:
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"regions": ["us-*", "ca-*"]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Lastly of course you can target all 14 regions currently supported:
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"regions": ["*"]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Multi-region support is still a work-in-progress as a few new AWS features are required to tie things together.
|
|
||||||
|
|
||||||
#### Static file serving
|
|
||||||
|
|
||||||
Up supports static file serving out of the box, with HTTP cache support, so you can use CloudFront or any other CDN in front of your application to dramatically reduce latency.
|
|
||||||
|
|
||||||
By default the working directory is served (`.`) when `type` is “static”, however you may provide a `static.dir` as well:
|
|
||||||
|
|
||||||
```
|
|
||||||
{ "name": "app", "type": "static", "static": { "dir": "public" }}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Build hooks
|
|
||||||
|
|
||||||
The build hooks allow you to define custom actions when deploying or performing other operations. A common example would be to bundle Node.js apps using Webpack or Browserify, greatly reducing the file size, as node_modules is _huge_ .
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"name": "app",
|
|
||||||
"hooks": {
|
|
||||||
"build": "browserify --node server.js > app.js",
|
|
||||||
"clean": "rm app.js"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Script and stylesheet injection
|
|
||||||
|
|
||||||
Up allows you to inject scripts and styles, either inline or paths in a declarative manner. It even supports a number of “canned” scripts for Google Analytics and [Segment][14], just copy & paste your write key.
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"name": "site",
|
|
||||||
"type": "static",
|
|
||||||
"inject": {
|
|
||||||
"head": [
|
|
||||||
{
|
|
||||||
"type": "segment",
|
|
||||||
"value": "API_KEY"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "inline style",
|
|
||||||
"file": "/css/primer.css"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"body": [
|
|
||||||
{
|
|
||||||
"type": "script",
|
|
||||||
"value": "/app.js"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Rewrites and redirects
|
|
||||||
|
|
||||||
Up supports redirects and URL rewriting via the `redirects` object, which maps path patterns to a new location. If `status` is omitted (or 200) then it is a rewrite, otherwise it is a redirect.
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"name": "app",
|
|
||||||
"type": "static",
|
|
||||||
"redirects": {
|
|
||||||
"/blog": {
|
|
||||||
"location": "https://blog.apex.sh/",
|
|
||||||
"status": 301
|
|
||||||
},
|
|
||||||
"/docs/:section/guides/:guide": {
|
|
||||||
"location": "/help/:section/:guide",
|
|
||||||
"status": 302
|
|
||||||
},
|
|
||||||
"/store/*": {
|
|
||||||
"location": "/shop/:splat"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
A common use-case for rewrites is for SPAs (Single Page Apps), where you want to serve the `index.html` file regardless of the path. Unless of course the file exists.
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"name": "app",
|
|
||||||
"type": "static",
|
|
||||||
"redirects": {
|
|
||||||
"/*": {
|
|
||||||
"location": "/",
|
|
||||||
"status": 200
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
If you want to force the rule regardless of a file existing, just add `"force": true` .
|
|
||||||
|
|
||||||
#### Environment variables
|
|
||||||
|
|
||||||
Secrets will be in the next release, however for now plain-text environment variables are supported:
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"name": "api",
|
|
||||||
"environment": {
|
|
||||||
"API_FEATURE_FOO": "1",
|
|
||||||
"API_FEATURE_BAR": "0"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### CORS support
|
|
||||||
|
|
||||||
The [CORS][16] support allows you to to specify which (if any) domains can access your API from the browser. If you wish to allow any site to access your API, just enable it:
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"cors": {
|
|
||||||
"enable": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
You can also customize access, for example restricting API access to your front-end or SPA only.
|
|
||||||
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"cors": {
|
|
||||||
"allowed_origins": ["https://myapp.com"],
|
|
||||||
"allowed_methods": ["HEAD", "GET", "POST", "PUT", "DELETE"],
|
|
||||||
"allowed_headers": ["Content-Type", "Authorization"]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Logging
|
|
||||||
|
|
||||||
For the low price of $0.5/GB you can utilize CloudWatch logs for structured log querying and tailing. Up implements a custom [query language][18] used to improve upon what CloudWatch provides, purpose-built for querying structured JSON logs.
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/2000/1*hrON4pH_WzN6CajaiU-ZYw.png)
|
|
||||||
|
|
||||||
You can query existing logs:
|
|
||||||
|
|
||||||
```
|
|
||||||
up logs
|
|
||||||
```
|
|
||||||
|
|
||||||
Tail live logs:
|
|
||||||
|
|
||||||
```
|
|
||||||
up logs -f
|
|
||||||
```
|
|
||||||
|
|
||||||
Or filter on either of them, for example only showing 200 GET / HEAD requests that take more than 5 milliseconds to complete:
|
|
||||||
|
|
||||||
```
|
|
||||||
up logs 'method in ("GET", "HEAD") status = 200 duration >= 5'
|
|
||||||
```
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/1600/1*Nhc5eiMM24gbiICFW7kBLg.png)
|
|
||||||
|
|
||||||
The query language is quite flexible, here are some more examples from `up help logs`
|
|
||||||
|
|
||||||
```
|
|
||||||
Show logs from the past 5 minutes.
|
|
||||||
$ up logs
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show logs from the past 30 minutes.
|
|
||||||
$ up logs -s 30m
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show logs from the past 5 hours.
|
|
||||||
$ up logs -s 5h
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show live log output.
|
|
||||||
$ up logs -f
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show error logs.
|
|
||||||
$ up logs error
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show error and fatal logs.
|
|
||||||
$ up logs 'error or fatal'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show non-info logs.
|
|
||||||
$ up logs 'not info'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show logs with a specific message.
|
|
||||||
$ up logs 'message = "user login"'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show 200 responses with latency above 150ms.
|
|
||||||
$ up logs 'status = 200 duration > 150'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show 4xx and 5xx responses.
|
|
||||||
$ up logs 'status >= 400'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show emails containing @apex.sh.
|
|
||||||
$ up logs 'user.email contains "@apex.sh"'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show emails ending with @apex.sh.
|
|
||||||
$ up logs 'user.email = "*@apex.sh"'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show emails starting with tj@.
|
|
||||||
$ up logs 'user.email = "tj@*"'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show errors from /tobi and /loki
|
|
||||||
$ up logs 'error and (path = "/tobi" or path = "/loki")'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show the same as above with 'in'
|
|
||||||
$ up logs 'error and path in ("/tobi", "/loki")'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Show logs with a more complex query.
|
|
||||||
$ up logs 'method in ("POST", "PUT") ip = "207.*" status = 200 duration >= 50'
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
Pipe JSON error logs to the jq tool.
|
|
||||||
$ up logs error | jq
|
|
||||||
```
|
|
||||||
|
|
||||||
Note that the `and` keyword is implied, though you can use it if you prefer.
|
|
||||||
|
|
||||||
#### Cold start times
|
|
||||||
|
|
||||||
This is a property of AWS Lambda as a platform, but the cold start times are typically well below 1 second, and in the future I plan on providing an option to keep them warm.
|
|
||||||
|
|
||||||
#### Config validation
|
|
||||||
|
|
||||||
The `up config` command outputs the resolved configuration, complete with defaults and inferred runtime settings – it also serves the dual purpose of validating configuration, as any error will result in exit > 0.
|
|
||||||
|
|
||||||
#### Crash recovery
|
|
||||||
|
|
||||||
Another benefit of using Up as a reverse proxy is performing crash recovery — restarting your server upon crashes and re-attempting the request before responding to the client with an error.
|
|
||||||
|
|
||||||
For example suppose your Node.js application crashes with an uncaught exception due to an intermittent database issue, Up can retry this request before ever responding to the client. Later this behaviour will be more customizable.
|
|
||||||
|
|
||||||
#### Continuous integration friendly
|
|
||||||
|
|
||||||
It’s hard to call this a feature, but thanks to Golang’s relatively small and isolated binaries, you can install Up in a CI in a second or two.
|
|
||||||
|
|
||||||
#### HTTP/2
|
|
||||||
|
|
||||||
Up supports HTTP/2 out of the box via API Gateway, reducing the latency for serving apps and sites with with many assets. I’ll do more comprehensive testing against many platforms in the future, but Up’s latency is already favourable:
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/1600/1*psg0kJND1UCryXEa0D3VBA.jpeg)
|
|
||||||
|
|
||||||
#### Error pages
|
|
||||||
|
|
||||||
Up provides a default error page which you may customize with `error_pages` if you’d like to provide a support email or tweak the color.
|
|
||||||
|
|
||||||
```
|
|
||||||
{ "name": "site", "type": "static", "error_pages": { "variables": { "support_email": "support@apex.sh", "color": "#228ae6" } }}
|
|
||||||
```
|
|
||||||
|
|
||||||
By default it looks like this:
|
|
||||||
|
|
||||||
** 此处有Canvas,请手动处理 **
|
|
||||||
|
|
||||||
![](https://cdn-images-1.medium.com/max/2000/1*_Mdj6uTCGvYTCoXsNOSD6w.png)
|
|
||||||
|
|
||||||
If you’d like to provide custom templates you may create one or more of the following files. The most specific file takes precedence.
|
|
||||||
|
|
||||||
* `error.html` – Matches any 4xx or 5xx
|
|
||||||
|
|
||||||
* `5xx.html` – Matches any 5xx error
|
|
||||||
|
|
||||||
* `4xx.html` – Matches any 4xx error
|
|
||||||
|
|
||||||
* `CODE.html` – Matches a specific code such as 404.html
|
|
||||||
|
|
||||||
Check out the [docs][22] to read more about templating.
|
|
||||||
|
|
||||||
### Scaling and cost
|
|
||||||
|
|
||||||
So you’ve made it this far, but how well does Up scale? Currently API Gateway and AWS are the target platform, so you’re not required to make any changes in order to scale, just deploy your code and it’s done. You pay only for what you actually use, on-demand, and no manual intervention is required for scaling.
|
|
||||||
|
|
||||||
AWS offers 1,000,000 requests per month for free, but you can use [http://serverlesscalc.com][23] to plug in your expected traffic. In the future Up will provide additional platforms, so that if one becomes prohibitively expensive, you can migrate to another!
|
|
||||||
|
|
||||||
### The Future
|
|
||||||
|
|
||||||
That’s all for now! It may not look like much, but it’s clocking-in above 10,000 lines of code already, and I’ve just begun development. Take a look at the issue queue for a small look at what to expect in the future, assuming the project becomes sustainable.
|
|
||||||
|
|
||||||
If you find the free version useful please consider donating on [OpenCollective][24], as I do not make any money working on it. I will be working on early access to the Pro version shortly, with a discounted annual price for early adopters. Either the Pro or Enterprise editions will provide the source as well, so internal hotfixes and customizations can be made.
|
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
via: https://medium.freecodecamp.org/up-b3db1ca930ee
|
|
||||||
|
|
||||||
作者:[TJ Holowaychuk ][a]
|
|
||||||
译者:[译者ID](https://github.com/译者ID)
|
|
||||||
校对:[校对者ID](https://github.com/校对者ID)
|
|
||||||
|
|
||||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
|
||||||
|
|
||||||
[a]:https://medium.freecodecamp.org/@tjholowaychuk?source=post_header_lockup
|
|
||||||
[1]:https://medium.com/@tjholowaychuk/blueprints-for-up-1-5f8197179275
|
|
||||||
[2]:https://github.com/apex/up
|
|
||||||
[3]:https://github.com/apex/up
|
|
||||||
[4]:https://github.com/tj/gh-polls
|
|
||||||
[5]:https://github.com/apex/up/tree/master/docs
|
|
||||||
[6]:https://github.com/apex/up/releases
|
|
||||||
[7]:https://raw.githubusercontent.com/apex/up/master/install.sh
|
|
||||||
[8]:https://github.com/apex/up/blob/master/docs/aws-credentials.md
|
|
||||||
[9]:https://github.com/apex/apex
|
|
||||||
[10]:https://github.com/apex/up/blob/master/docs/runtimes.md
|
|
||||||
[11]:https://github.com/apex/up/issues/166
|
|
||||||
[12]:https://github.com/apex/up/issues/115
|
|
||||||
[13]:https://github.com/apex/up/blob/master/docs/configuration.md
|
|
||||||
[14]:https://segment.com/
|
|
||||||
[15]:https://blog.apex.sh/
|
|
||||||
[16]:https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS
|
|
||||||
[17]:https://myapp.com/
|
|
||||||
[18]:https://github.com/apex/up/blob/master/internal/logs/parser/grammar.peg
|
|
||||||
[19]:http://twitter.com/apex
|
|
||||||
[20]:http://twitter.com/apex
|
|
||||||
[21]:http://twitter.com/apex
|
|
||||||
[22]:https://github.com/apex/up/blob/master/docs/configuration.md#error-pages
|
|
||||||
[23]:http://serverlesscalc.com/
|
|
||||||
[24]:https://opencollective.com/apex-up
|
|
@ -0,0 +1,539 @@
|
|||||||
|
UP - 在几秒钟内部署无服务器应用程序
|
||||||
|
============================================================
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/2000/1*8KijrYCm1j0_XvrACQD_fQ.png)
|
||||||
|
|
||||||
|
去年,我[为 Up 写了一份蓝图][1],其中描述了大多数构建块是如何以最小的成本在 AWS 上创建一个很棒的无服务器体验。这篇文章谈到了 [Up][2] 的初始 alpha 版本。
|
||||||
|
|
||||||
|
为什么专注于无服务器?对于初学者来说,它可以节省成本,因为你可以按需付费,且只为你使用的付费。无服务器选项是自我修复的,因为每个请求被隔离并被认为是“无状态的”。最后,它可以无限轻松扩展 - 没有机器或集群要管理。部署你的代码就完成了。
|
||||||
|
|
||||||
|
大约一个月前,我决定使用 [apex/up][3],并为在线 SVG GitHub 用户调查写了第一个小型无服务器示例程序 [tj/gh-polls][4]。它运行良好,成本低于每月 1 美元,为数百万调查服务,因此我会继续这个项目,看看我是否可以提供开源和商业的变体。
|
||||||
|
|
||||||
|
长期的目标是提供“你自己即 Heroku” 的版本,支持许多平台。虽然平台即服务并不新鲜,但无服务器生态系统正在使这种方案日益微不足道。据说,AWS 和其他的经常因为 UX 提供的灵活性而被人诟病。Up 将复杂性抽象出来,同时为你提供一个几乎无需操作的解决方案。
|
||||||
|
|
||||||
|
### 安装
|
||||||
|
|
||||||
|
你可以使用以下命令安装 Up,查看[临时文档][5]开始使用。或者如果你使用安装脚本,请下载[二进制版本][6]。(请记住,这个项目还在早期。)
|
||||||
|
|
||||||
|
```
|
||||||
|
curl -sfL https://raw.githubusercontent.com/apex/up/master/install.sh | sh
|
||||||
|
```
|
||||||
|
|
||||||
|
只需运行以下命令随时升级到最新版本:
|
||||||
|
|
||||||
|
```
|
||||||
|
up upgrade
|
||||||
|
```
|
||||||
|
|
||||||
|
你也可以通过NPM进行安装:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm install -g up
|
||||||
|
```
|
||||||
|
|
||||||
|
### 功能
|
||||||
|
|
||||||
|
早期 alpha 提供什么功能?让我们来看看!请记住,Up 不是托管服务,因此你需要一个 AWS 帐户和[ AWS 凭证][8]。如果你对 AWS 不熟悉,你可能需要先停下直到熟悉流程。
|
||||||
|
|
||||||
|
我的第一个问题是:up(1) 与 [apex(1)][9] 有何不同?Apex 专注于部署功能,用于管道和事件处理,而 Up 则侧重于应用程序、apis 和静态站点,也就是单个可部署单元。Apex 不为你提供 API 网关、SSL 证书或 DNS,也不提供 URL 重写,脚本注入等。
|
||||||
|
|
||||||
|
#### 单命令无服务器应用程序
|
||||||
|
|
||||||
|
Up 可以让你使用单条命令部署应用程序、apis 和静态站点。要创建一个应用程序,你需要的是一个文件,在 Node.js 的情况下,`./app.js` 监听由 Up 提供的 `PORT'。请注意,如果你使用的是 `package.json` ,则会检测并使用 `start`和 `build` 脚本。
|
||||||
|
|
||||||
|
```
|
||||||
|
const http = require('http')
|
||||||
|
const { PORT = 3000 } = process.env
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
http.createServer((req, res) => {
|
||||||
|
res.end('Hello World\n')
|
||||||
|
}).listen(PORT)
|
||||||
|
```
|
||||||
|
|
||||||
|
额外的[运行时][10]支持开箱即用,例如 Golang 的“main.go”,所以你可以在几秒钟内部署 Golang、Python、Crystal 或 Node.js 应用程序。
|
||||||
|
|
||||||
|
```
|
||||||
|
package main
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
func main() {
|
||||||
|
addr := ":" + os.Getenv("PORT")
|
||||||
|
http.HandleFunc("/", hello)
|
||||||
|
log.Fatal(http.ListenAndServe(addr, nil))
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
func hello(w http.ResponseWriter, r *http.Request) {
|
||||||
|
fmt.Fprintln(w, "Hello World from Go")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
要部署应用程序输入 `up` 来创建所需的资源,并部署应用程序本身。这里没有迷雾,一旦它说“完成”了,你就完成了,该应用程序立即可用 - 没有远程构建过程。
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/2000/1*tBYR5HXeDDVkb_Pv2MCj1A.png)
|
||||||
|
|
||||||
|
后续的部署将会更快,因为栈已被配置:
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/2000/1*2w2WHDTfTT-7GsMtNPklXw.png)
|
||||||
|
|
||||||
|
使用 `up url --open` 测试你的程序,以在浏览器中浏览它,`up url --copy` 将 URL 保存到剪贴板,或者尝试使用 curl:
|
||||||
|
|
||||||
|
```
|
||||||
|
curl `up url`
|
||||||
|
Hello World
|
||||||
|
```
|
||||||
|
|
||||||
|
To delete the app and its resources just type `up stack delete`:
|
||||||
|
要删除应用程序及其资源,只需输入 `up stack delete`:
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/2000/1*FUdhBTtDHaZ2CEPHR7PGqg.png)
|
||||||
|
|
||||||
|
例如,使用 `up staging` 或 `up production` 和 `up url --open production` 部署到预发布或生产环境。请注意,自定义域名尚不可用,[它们将很快可用][11]。之后,你还可以将版本“推广”到其他环境。
|
||||||
|
|
||||||
|
#### 反向代理
|
||||||
|
|
||||||
|
一个使 Up 独特的功能是,它不仅仅是简单地部署代码,它将一个 Golang 反向代理放在应用程序的前面。这提供了许多功能,如 URL 重写、重定向、脚本注入等等,我们将在后面进一步介绍。
|
||||||
|
|
||||||
|
#### 基础设施即代码
|
||||||
|
|
||||||
|
在配置方面,Up 遵循现代最佳实践,因此多有对基础设施的更改都可以在部署之前预览,并且 IAM 策略的使用还可以限制开发人员访问以防止事故发生。一个好处是它有助于自动记录你的基础设施。
|
||||||
|
|
||||||
|
以下是使用 LetsEncrypt 通过 AWS ACM 配置一些(虚拟)DNS 记录和免费 SSL 证书的示例。
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "app",
|
||||||
|
"dns": {
|
||||||
|
"myapp.com": [
|
||||||
|
{
|
||||||
|
"name": "myapp.com",
|
||||||
|
"type": "A",
|
||||||
|
"ttl": 300,
|
||||||
|
"value": ["35.161.83.243"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "blog.myapp.com",
|
||||||
|
"type": "CNAME",
|
||||||
|
"ttl": 300,
|
||||||
|
"value": ["34.209.172.67"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "api.myapp.com",
|
||||||
|
"type": "A",
|
||||||
|
"ttl": 300,
|
||||||
|
"value": ["54.187.185.18"]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"certs": [
|
||||||
|
{
|
||||||
|
"domains": ["myapp.com", "*.myapp.com"]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
当你首次通过 `up` 部署应用程序时,需要所有的权限,它为你创建 API 网关、Lambda 函数、ACM 证书、Route53 DNS 记录等。
|
||||||
|
|
||||||
|
[ChangeSets][12] 尚未实现,但你能使用 `up stack plan` 预览进一步的更改,并使用 `up stack apply` 提交,这与 Terraform 非常相似。
|
||||||
|
|
||||||
|
详细信息请参阅[配置文档][13]。
|
||||||
|
|
||||||
|
#### 全球部署
|
||||||
|
|
||||||
|
`regions` 数组可以指定应用程序的目标区域。例如,如果你只对单个地区感兴趣,请使用:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"regions": ["us-west-2"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你的客户集中在北美,你可能需要使用美国和加拿大所有地区:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"regions": ["us-*", "ca-*"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
最后,你可以使用目前支持的所有 14 个地区:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"regions": ["*"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
多区域支持仍然是一个正在进行的工作,因为需要一些新的 AWS 功能来将它们结合在一起。
|
||||||
|
|
||||||
|
#### 静态文件服务
|
||||||
|
|
||||||
|
Up 开箱即支持静态文件服务,支持 HTTP 缓存,因此你可以在应用程序前使用 CloudFront 或任何其他 CDN 来大大减少延迟。
|
||||||
|
|
||||||
|
当 `type` 为 “static” 时,默认情况下的工作目录是(`.`),但是你也可以提供一个`static.dir`:
|
||||||
|
|
||||||
|
```
|
||||||
|
{ "name": "app", "type": "static", "static": { "dir": "public" }}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 构建钩子
|
||||||
|
|
||||||
|
构建钩子允许你在部署或执行其他操作时定义自定义操作。一个常见的例子是使用 Webpack 或 Browserify 捆绑 Node.js 应用程序,这大大减少了文件大小,因为 node_modules 是_很大_的。
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "app",
|
||||||
|
"hooks": {
|
||||||
|
"build": "browserify --node server.js > app.js",
|
||||||
|
"clean": "rm app.js"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 脚本和样式表注入
|
||||||
|
|
||||||
|
Up 允许你插入脚本和样式,它可以内联或声明路径。它甚至支持一些“罐头”脚本,用于 Google Analytics(分析)和 [Segment][14],只需复制并粘贴你的写入密钥即可。
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "site",
|
||||||
|
"type": "static",
|
||||||
|
"inject": {
|
||||||
|
"head": [
|
||||||
|
{
|
||||||
|
"type": "segment",
|
||||||
|
"value": "API_KEY"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "inline style",
|
||||||
|
"file": "/css/primer.css"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"body": [
|
||||||
|
{
|
||||||
|
"type": "script",
|
||||||
|
"value": "/app.js"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 重写和重定向
|
||||||
|
|
||||||
|
Up通过 `redirects` 对象支持重定向和 URL 重写,该对象将路径模式映射到新位置。如果省略 `status`(或200),那么它是重写,否则是重定向。
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "app",
|
||||||
|
"type": "static",
|
||||||
|
"redirects": {
|
||||||
|
"/blog": {
|
||||||
|
"location": "https://blog.apex.sh/",
|
||||||
|
"status": 301
|
||||||
|
},
|
||||||
|
"/docs/:section/guides/:guide": {
|
||||||
|
"location": "/help/:section/:guide",
|
||||||
|
"status": 302
|
||||||
|
},
|
||||||
|
"/store/*": {
|
||||||
|
"location": "/shop/:splat"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
用于重写的常见情况是 SPA(单页面应用程序),你希望为 `index.html` 提供服务,而不管路径如何。当然除非文件存在。
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "app",
|
||||||
|
"type": "static",
|
||||||
|
"redirects": {
|
||||||
|
"/*": {
|
||||||
|
"location": "/",
|
||||||
|
"status": 200
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
如果要强制规则,无论文件是否存在,只需添加 `"force": true` 。
|
||||||
|
|
||||||
|
#### 环境变量
|
||||||
|
|
||||||
|
密码将在下一个版本中有,但是现在支持纯文本环境变量:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"name": "api",
|
||||||
|
"environment": {
|
||||||
|
"API_FEATURE_FOO": "1",
|
||||||
|
"API_FEATURE_BAR": "0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### CORS 支持
|
||||||
|
|
||||||
|
[CORS][16] 支持允许你指定哪些(如果有的话)域可以从浏览器访问你的 API。如果你希望允许任何网站访问你的 API,只需启用它:
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"cors": {
|
||||||
|
"enable": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
你还可以自定义访问,例如仅限制 API 访问你的前端或 SPA。
|
||||||
|
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"cors": {
|
||||||
|
"allowed_origins": ["https://myapp.com"],
|
||||||
|
"allowed_methods": ["HEAD", "GET", "POST", "PUT", "DELETE"],
|
||||||
|
"allowed_headers": ["Content-Type", "Authorization"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 日志
|
||||||
|
|
||||||
|
对于 $0.5/GB 的低价格,你可以使用 CloudWatch 日志进行结构化日志查询和跟踪。Up 实现了一种用于改进 CloudWatch 提供的自定义[查询语言][18],专门用于查询结构化 JSON 日志。
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/2000/1*hrON4pH_WzN6CajaiU-ZYw.png)
|
||||||
|
|
||||||
|
你可以查询现有日志:
|
||||||
|
|
||||||
|
```
|
||||||
|
up logs
|
||||||
|
```
|
||||||
|
|
||||||
|
跟踪在线日志:
|
||||||
|
|
||||||
|
```
|
||||||
|
up logs -f
|
||||||
|
```
|
||||||
|
|
||||||
|
或者对其中任一个进行过滤,例如只显示耗时超过 5 毫秒的 200 个 GET/HEAD 请求:
|
||||||
|
|
||||||
|
```
|
||||||
|
up logs 'method in ("GET", "HEAD") status = 200 duration >= 5'
|
||||||
|
```
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/1600/1*Nhc5eiMM24gbiICFW7kBLg.png)
|
||||||
|
|
||||||
|
查询语言是非常灵活的,这里有更多来自于 `up help logs` 的例子
|
||||||
|
|
||||||
|
```
|
||||||
|
Show logs from the past 5 minutes.
|
||||||
|
$ up logs
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show logs from the past 30 minutes.
|
||||||
|
$ up logs -s 30m
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show logs from the past 5 hours.
|
||||||
|
$ up logs -s 5h
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show live log output.
|
||||||
|
$ up logs -f
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show error logs.
|
||||||
|
$ up logs error
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show error and fatal logs.
|
||||||
|
$ up logs 'error or fatal'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show non-info logs.
|
||||||
|
$ up logs 'not info'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show logs with a specific message.
|
||||||
|
$ up logs 'message = "user login"'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show 200 responses with latency above 150ms.
|
||||||
|
$ up logs 'status = 200 duration > 150'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show 4xx and 5xx responses.
|
||||||
|
$ up logs 'status >= 400'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show emails containing @apex.sh.
|
||||||
|
$ up logs 'user.email contains "@apex.sh"'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show emails ending with @apex.sh.
|
||||||
|
$ up logs 'user.email = "*@apex.sh"'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show emails starting with tj@.
|
||||||
|
$ up logs 'user.email = "tj@*"'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show errors from /tobi and /loki
|
||||||
|
$ up logs 'error and (path = "/tobi" or path = "/loki")'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show the same as above with 'in'
|
||||||
|
$ up logs 'error and path in ("/tobi", "/loki")'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Show logs with a more complex query.
|
||||||
|
$ up logs 'method in ("POST", "PUT") ip = "207.*" status = 200 duration >= 50'
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
Pipe JSON error logs to the jq tool.
|
||||||
|
$ up logs error | jq
|
||||||
|
```
|
||||||
|
|
||||||
|
请注意,`and` 关键字是暗含的,虽然你也可以使用它。
|
||||||
|
|
||||||
|
#### 冷启动时间
|
||||||
|
|
||||||
|
这是 AWS Lambda 平台的特性, 但冷启动时间通常远远低于 1 秒, 在未来, 我计划提供一个选项来保持它们在线。
|
||||||
|
|
||||||
|
#### 配置验证
|
||||||
|
|
||||||
|
The `up config` command outputs the resolved configuration, complete with defaults and inferred runtime settings – it also serves the dual purpose of validating configuration, as any error will result in exit > 0.
|
||||||
|
`up config` 命令输出解析后的配置,有默认值和推断的运行时设置 - 它也起到验证配置的双重目的,因为任何错误都会导致 exit > 0。
|
||||||
|
|
||||||
|
#### 崩溃恢复
|
||||||
|
|
||||||
|
使用 Up 作为反向代理的另一个好处是执行崩溃恢复 - 在崩溃后重新启动服务器,并在响应客户端发生错误之前重新尝试该请求。
|
||||||
|
|
||||||
|
例如,假设你的 Node.js 程序由于间歇性数据库问题而导致未捕获的异常崩溃,Up 可以在响应客户端之前重试该请求。之后这个行为会更加可定制。
|
||||||
|
|
||||||
|
#### 持续集成友好
|
||||||
|
|
||||||
|
很难说这是一个功能,但是感谢 Golang 相对较小和独立的二进制文件,你可以在一两秒中在 CI 中安装 Up。
|
||||||
|
|
||||||
|
#### HTTP/2
|
||||||
|
|
||||||
|
Up 通过 API 网关支持 HTTP/2,对服务有很多资源的应用和站点减少延迟。我将来会对许多平台进行更全面的测试,但是 Up 的延迟已经很好了:
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/1600/1*psg0kJND1UCryXEa0D3VBA.jpeg)
|
||||||
|
|
||||||
|
#### 错误页面
|
||||||
|
|
||||||
|
Up 提供了一个默认错误页面,如果你要提供支持电子邮件或调整颜色,你可以使用 `error_pages` 自定义。
|
||||||
|
|
||||||
|
```
|
||||||
|
{ "name": "site", "type": "static", "error_pages": { "variables": { "support_email": "support@apex.sh", "color": "#228ae6" } }}
|
||||||
|
```
|
||||||
|
|
||||||
|
默认情况下,它看上去像这样:
|
||||||
|
|
||||||
|
** 此处有Canvas,请手动处理 **
|
||||||
|
|
||||||
|
![](https://cdn-images-1.medium.com/max/2000/1*_Mdj6uTCGvYTCoXsNOSD6w.png)
|
||||||
|
|
||||||
|
如果你想提供自定义模板,你可以创建以下一个或多个文件。特定文件优先。
|
||||||
|
|
||||||
|
* `error.html` – 匹配任何 4xx 或 5xx
|
||||||
|
|
||||||
|
* `5xx.html` – 匹配任何 5xx 错误
|
||||||
|
|
||||||
|
* `4xx.html` – 匹配任何 4xx 错误
|
||||||
|
|
||||||
|
* `CODE.html` – 匹配一个特定的代码,如 404.html
|
||||||
|
|
||||||
|
查看[文档][22]阅读更多有关模板的信息。
|
||||||
|
|
||||||
|
### 伸缩和成本
|
||||||
|
|
||||||
|
你已经做了这么多,但是 Up 规模如何?目前,API 网关和 AWS 是目标平台,因此你无需进行任何更改即可扩展,只需部署代码即可完成。你只需支付实际使用的数量、按需并且无需人工干预。
|
||||||
|
|
||||||
|
AWS 每月免费提供 1,000,000 个请求,但你可以使用 [http://serverlesscalc.com][23] 来插入预期流量。在未来 Up 将提供额外的平台,所以如果一个成本过高,你可以迁移到另一个!
|
||||||
|
|
||||||
|
### 未来
|
||||||
|
|
||||||
|
目前为止就这样了!它可能看起来不是很多,但它已经超过 10,000 行代码,并且我刚刚开始开发。看看这个问题队列,假设项目可持续发展,看看未来会有什么期待。
|
||||||
|
|
||||||
|
如果你发现免费版本有用,请考虑在 [OpenCollective][24] 上捐赠 ,因为我没有任何工作。我将在短期内开发早期专业版,早期用户的年费优惠。专业或企业版也将提供源码,因此可以进行内部修复和自定义。
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
via: https://medium.freecodecamp.org/up-b3db1ca930ee
|
||||||
|
|
||||||
|
作者:[TJ Holowaychuk ][a]
|
||||||
|
译者:[geekpi](https://github.com/geekpi)
|
||||||
|
校对:[校对者ID](https://github.com/校对者ID)
|
||||||
|
|
||||||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||||
|
|
||||||
|
[a]:https://medium.freecodecamp.org/@tjholowaychuk?source=post_header_lockup
|
||||||
|
[1]:https://medium.com/@tjholowaychuk/blueprints-for-up-1-5f8197179275
|
||||||
|
[2]:https://github.com/apex/up
|
||||||
|
[3]:https://github.com/apex/up
|
||||||
|
[4]:https://github.com/tj/gh-polls
|
||||||
|
[5]:https://github.com/apex/up/tree/master/docs
|
||||||
|
[6]:https://github.com/apex/up/releases
|
||||||
|
[7]:https://raw.githubusercontent.com/apex/up/master/install.sh
|
||||||
|
[8]:https://github.com/apex/up/blob/master/docs/aws-credentials.md
|
||||||
|
[9]:https://github.com/apex/apex
|
||||||
|
[10]:https://github.com/apex/up/blob/master/docs/runtimes.md
|
||||||
|
[11]:https://github.com/apex/up/issues/166
|
||||||
|
[12]:https://github.com/apex/up/issues/115
|
||||||
|
[13]:https://github.com/apex/up/blob/master/docs/configuration.md
|
||||||
|
[14]:https://segment.com/
|
||||||
|
[15]:https://blog.apex.sh/
|
||||||
|
[16]:https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS
|
||||||
|
[17]:https://myapp.com/
|
||||||
|
[18]:https://github.com/apex/up/blob/master/internal/logs/parser/grammar.peg
|
||||||
|
[19]:http://twitter.com/apex
|
||||||
|
[20]:http://twitter.com/apex
|
||||||
|
[21]:http://twitter.com/apex
|
||||||
|
[22]:https://github.com/apex/up/blob/master/docs/configuration.md#error-pages
|
||||||
|
[23]:http://serverlesscalc.com/
|
||||||
|
[24]:https://opencollective.com/apex-up
|
Loading…
Reference in New Issue
Block a user