Go作为构建Web应用程序的首选语言正变得越来越流行。
这在很大程度上要归功于它的速度,应用程序性能以及可移植性。互联网上有很多资源可以教您如何在Go中构建端到端Web应用程序,但是在大多数情况下,它们要么散布在孤立的博客文章中,要么散布在书籍中。
这里将从0开始搭建一个完整的项目。
项目效果
我们将建立鸟类社区百科全书。该网站将:
- 显示社区提交的不同条目,以及它们找到的鸟的名称和详细信息。
- 允许任何人张贴关于他们看见的鸟的新条目。
此应用程序将需要三个组件:
- Web服务器
- 前端(客户端)应用
- 数据库
设置环境
配置GOPATH,就是将GOPATH添加到系统的环境变量中去。
设置目录
gopath目录设置三个子目录,没有创建,分别是src,bin,pkg
- src 所有 Go源代码的去向,包括我们自己要创建的代码
- bin go编译之后的可执行文件
- pkg 包含由库制作的包对象(您现在不必担心)
创建项目目录
src目录中的项目文件夹应遵循与远程存储库所在位置相同的位置结构。因此,例如,如果我要创建一个名为"birdpedia"的新项目,并在github上以我的名字创建该项目的存储库,这样我的项目存储库的位置将在"github.com/dream2012/birdwiki"上,则该项目在我的计算机上的位置为:$GOPATH/src/github.com/dream2012/birdwiki
启动http服务器
现在几乎所有语言都实现了内置的服务器功能,golang亦是如此。
创建一个main.go
// This is the name of our package
// Everything with this package name can see everything
// else inside the same package, regardless of the file they are in
package main
// These are the libraries we are going to use
// Both "fmt" and "net" are part of the Go standard library
import (
// "fmt" has methods for formatted I/O operations (like printing to the console)
"fmt"
// The "net/http" library has methods to implement HTTP clients and servers
"net/http"
)
func main() {
// The "HandleFunc" method accepts a path and a function as arguments
// (Yes, we can pass functions as arguments, and even trat them like variables in Go)
// However, the handler function has to have the appropriate signature (as described by the "handler" function below)
http.HandleFunc("/", handler)
// After defining our server, we finally "listen and serve" on port 8080
// The second argument is the handler, which we will come to later on, but for now it is left as nil,
// and the handler defined above (in "HandleFunc") is used
http.ListenAndServe(":8080", nil)
}
// "handler" is our handler function. It has to follow the function signature of a ResponseWriter and Request type
// as the arguments.
func handler(w http.ResponseWriter, r *http.Request) {
// For this case, we will always pipe "Hello World" into the response writer
fmt.Fprintf(w, "Hello World!")
}
go run main.go
我们打开浏览器,访问http://127.0.0.1:8080/ 就可以看见"Hello Wolrd!"
设置路由
现在我们无论如何修改url,看到的都是hello world,这是因为我们没有设置路由,就是根据不同url进行不同的操作处理。
安装mux路由
go get github.com/gorilla/mux
package main
import (
// Import the gorilla/mux library we just installed
"fmt"
"net/http"
"github.com/gorilla/mux"
)
func main() {
// Declare a new router
r := mux.NewRouter()
// This is where the router is useful, it allows us to declare methods that
// this path will be valid for
r.HandleFunc("/hello", handler).Methods("GET")
// We can then pass our router (after declaring all our routes) to this method
// (where previously, we were leaving the secodn argument as nil)
http.ListenAndServe(":8080", r)
}
func handler(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, "Hello World!")
}
这回我们只有访问http://127.0.0.1/hello 才能看到hello world,其它地址会看到404
测试
好的项目是需要有测试的,这里我们新建一个测试文件1_test.go
这里命名规则采取*_test.go
当我们执行go test 的时候,它会自动搜索_test.go文件进行测试
package main
import (
"net/http"
"net/http/httptest"
"testing"
)
func TestHandler(t *testing.T) {
//Here, we form a new HTTP request. This is the request that's going to be
// passed to our handler.
// The first argument is the method, the second argument is the route (which
//we leave blank for now, and will get back to soon), and the third is the
//request body, which we don't have in this case.
req, err := http.NewRequest("GET", "", nil)
// In case there is an error in forming the request, we fail and stop the test
if err != nil {
t.Fatal(err)
}
// We use Go's httptest library to create an http recorder. This recorder
// will act as the target of our http request
// (you can think of it as a mini-browser, which will accept the result of
// the http request that we make)
recorder := httptest.NewRecorder()
// Create an HTTP handler from our handler function. "handler" is the handler
// function defined in our main.go file that we want to test
hf := http.HandlerFunc(handler)
// Serve the HTTP request to our recorder. This is the line that actually
// executes our the handler that we want to test
hf.ServeHTTP(recorder, req)
// Check the status code is what we expect.
if status := recorder.Code; status != http.StatusOK {
t.Errorf("handler returned wrong status code: got %v want %v",
status, http.StatusOK)
}
// Check the response body is what we expect.
expected := `Hello World!`
actual := recorder.Body.String()
if actual != expected {
t.Errorf("handler returned unexpected body: got %v want %v", actual, expected)
}
}
静态文件
“静态文件”是构成网站所需的HTML,CSS,JavaScript,图像和其他静态资产文件。
为了使我们的服务器为这些静态资产提供服务,我们需要执行3个步骤。
- 创建静态文件
- 修改我们的路由器以提供静态文件
- 添加测试以验证我们的新服务器可以提供静态文件
创建static文件夹,创建静态文件index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<h1>staic html file</h1>
</body>
</html>
// This is the name of our package
// Everything with this package name can see everything
// else inside the same package, regardless of the file they are in
package main
// These are the libraries we are going to use
// Both "fmt" and "net" are part of the Go standard library
import (
// "fmt" has methods for formatted I/O operations (like printing to the console)
"fmt"
// The "net/http" library has methods to implement HTTP clients and servers
"net/http"
"github.com/gorilla/mux"
)
func main() {
// import router
r := mux.NewRouter()
r.HandleFunc("/hello", handler).Methods("GET")
staticFileDirectory := http.Dir("./static/")
staticFileHandler := http.StripPrefix("/static/", http.FileServer(staticFileDirectory))
// The "PathPrefix" method acts as a matcher, and matches all routes starting
// with "/assets/", instead of the absolute route itself
r.PathPrefix("/static/").Handler(staticFileHandler).Methods("GET")
// The "HandleFunc" method accepts a path and a function as arguments
// (Yes, we can pass functions as arguments, and even trat them like variables in Go)
// However, the handler function has to have the appropriate signature (as described by the "handler" function below)
//http.HandleFunc("/", handler)
// After defining our server, we finally "listen and serve" on port 8080
// The second argument is the handler, which we will come to later on, but for now it is left as nil,
// and the handler defined above (in "HandleFunc") is used
http.ListenAndServe(":8080", r)
}
// "handler" is our handler function. It has to follow the function signature of a ResponseWriter and Request type
// as the arguments.
func handler(w http.ResponseWriter, r *http.Request) {
// For this case, we will always pipe "Hello World" into the response writer
fmt.Fprintf(w, "Hello World!")
}
package main
import (
"net/http"
"net/http/httptest"
"testing"
"github.com/gorilla/mux"
)
func newRouter() *mux.Router {
r := mux.NewRouter()
r.HandleFunc("/hello", handler).Methods("GET")
// Declare the static file directory and point it to the
// directory we just made
staticFileDirectory := http.Dir("./static/")
staticFileHandler := http.StripPrefix("/static/", http.FileServer(staticFileDirectory))
r.PathPrefix("/static/").Handler(staticFileHandler).Methods("GET")
return r
}
func TestStaticFileServer(t *testing.T) {
r := newRouter()
mockServer := httptest.NewServer(r)
// We want to hit the `GET /assets/` route to get the index.html file response
resp, err := http.Get(mockServer.URL + "/static/")
if err != nil {
t.Fatal(err)
}
// We want our status to be 200 (ok)
if resp.StatusCode != http.StatusOK {
t.Errorf("Status should be 200, got %d", resp.StatusCode)
}
// It isn't wise to test the entire content of the HTML file.
// Instead, we test that the content-type header is "text/html; charset=utf-8"
// so that we know that an html file has been served
contentType := resp.Header.Get("Content-Type")
expectedContentType := "text/html; charset=utf-8"
if expectedContentType != contentType {
t.Errorf("Wrong content type, expected %s, got %s", expectedContentType, contentType)
}
}
创建表单 声明对象
golang中我们使用最后标识变量类型例如
type Bird struct{
Species string `json:"species"`
Description string `json:"description"`
}
<!DOCTYPE html>
<html lang="en">
<head>
<title>The bird encyclopedia</title>
</head>
<body>
<h1>The bird encyclopedia</h1>
<!--
This section of the document specifies the table that will
be used to display the list of birds and their description
-->
<table>
<tr>
<th>Species</th>
<th>Description</th>
</tr>
<td>Pigeon</td>
<td>Common in cities</td>
</tr>
</table>
<br/>
<!--
This section contains the form, that will be used to hit the
`POST /bird` API that we will build in the next section
-->
<form action="/bird" method="post">
Species:
<input type="text" name="species">
<br/> Description:
<input type="text" name="description">
<br/>
<input type="submit" value="Submit">
</form>
<!--
Finally, the last section is the script that will
run on each page load to fetch the list of birds
and add them to our existing table
-->
<script>
birdTable = document.querySelector("table")
/*
Use the browsers `fetch` API to make a GET call to /bird
We expect the response to be a JSON list of birds, of the
form :
[
{"species":"...","description":"..."},
{"species":"...","description":"..."}
]
*/
fetch("/bird")
.then(response => response.json())
.then(birdList => {
//Once we fetch the list, we iterate over it
birdList.forEach(bird => {
// Create the table row
row = document.createElement("tr")
// Create the table data elements for the species and
// description columns
species = document.createElement("td")
species.innerHTML = bird.species
description = document.createElement("td")
description.innerHTML = bird.description
// Add the data elements to the row
row.appendChild(species)
row.appendChild(description)
// Finally, add the row element to the table itself
birdTable.appendChild(row)
})
})
</script>
</body>
</html>
处理程序
// This is the name of our package
// Everything with this package name can see everything
// else inside the same package, regardless of the file they are in
package main
// These are the libraries we are going to use
// Both "fmt" and "net" are part of the Go standard library
import (
// "fmt" has methods for formatted I/O operations (like printing to the console)
"encoding/json"
"fmt"
// The "net/http" library has methods to implement HTTP clients and servers
"net/http"
"github.com/gorilla/mux"
)
type Bird struct {
Species string `json:"species"`
Description string `json:"description"`
}
var birds []Bird
func getBirdHandler(w http.ResponseWriter, r *http.Request) {
//Convert the "birds" variable to json
birdListBytes, err := json.Marshal(birds)
// If there is an error, print it to the console, and return a server
// error response to the user
if err != nil {
fmt.Println(fmt.Errorf("Error: %v", err))
w.WriteHeader(http.StatusInternalServerError)
return
}
// If all goes well, write the JSON list of birds to the response
w.Write(birdListBytes)
}
func createBirdHandler(w http.ResponseWriter, r *http.Request) {
// Create a new instance of Bird
bird := Bird{}
// We send all our data as HTML form data
// the `ParseForm` method of the request, parses the
// form values
err := r.ParseForm()
// In case of any error, we respond with an error to the user
if err != nil {
fmt.Println(fmt.Errorf("Error: %v", err))
w.WriteHeader(http.StatusInternalServerError)
return
}
// Get the information about the bird from the form info
bird.Species = r.Form.Get("species")
bird.Description = r.Form.Get("description")
// Append our existing list of birds with a new entry
birds = append(birds, bird)
//Finally, we redirect the user to the original HTMl page
// (located at `/assets/`), using the http libraries `Redirect` method
http.Redirect(w, r, "/assets/", http.StatusFound)
}
func main() {
// import router
r := mux.NewRouter()
r.HandleFunc("/hello", handler).Methods("GET")
staticFileDirectory := http.Dir("./static/")
staticFileHandler := http.StripPrefix("/static/", http.FileServer(staticFileDirectory))
// The "PathPrefix" method acts as a matcher, and matches all routes starting
// with "/assets/", instead of the absolute route itself
r.PathPrefix("/static/").Handler(staticFileHandler).Methods("GET")
r.HandleFunc("/bird", getBirdHandler).Methods("GET")
r.HandleFunc("/bird", createBirdHandler).Methods("POST")
// The "HandleFunc" method accepts a path and a function as arguments
// (Yes, we can pass functions as arguments, and even trat them like variables in Go)
// However, the handler function has to have the appropriate signature (as described by the "handler" function below)
//http.HandleFunc("/", handler)
// After defining our server, we finally "listen and serve" on port 8080
// The second argument is the handler, which we will come to later on, but for now it is left as nil,
// and the handler defined above (in "HandleFunc") is used
http.ListenAndServe(":8080", r)
}
// "handler" is our handler function. It has to follow the function signature of a ResponseWriter and Request type
// as the arguments.
func handler(w http.ResponseWriter, r *http.Request) {
// For this case, we will always pipe "Hello World" into the response writer
fmt.Fprintf(w, "Hello World!")
}
到此我们创建了一个简单wiki,我们可以自己添加数据,展示数据,剩下的就是如何保存数据到数据库了额,这个我们下回再说。