作者 yangfu

fix

要显示太多修改。

为保证性能只显示 21 of 21+ 个文件。

1 -.idea  
2 -.vscode  
3 -.DS_Store  
4 -*.swp  
5 -*.swo  
6 -beego.iml  
1 -github.com/astaxie/beego/*/*:S1012  
2 -github.com/astaxie/beego/*:S1012  
3 -github.com/astaxie/beego/*/*:S1007  
4 -github.com/astaxie/beego/*:S1007  
1 -language: go  
2 -  
3 -go:  
4 - - "1.9.2"  
5 - - "1.10.3"  
6 -services:  
7 - - redis-server  
8 - - mysql  
9 - - postgresql  
10 - - memcached  
11 -env:  
12 - - ORM_DRIVER=sqlite3 ORM_SOURCE=$TRAVIS_BUILD_DIR/orm_test.db  
13 - - ORM_DRIVER=postgres ORM_SOURCE="user=postgres dbname=orm_test sslmode=disable"  
14 -before_install:  
15 - - git clone git://github.com/ideawu/ssdb.git  
16 - - cd ssdb  
17 - - make  
18 - - cd ..  
19 -install:  
20 - - go get github.com/lib/pq  
21 - - go get github.com/go-sql-driver/mysql  
22 - - go get github.com/mattn/go-sqlite3  
23 - - go get github.com/bradfitz/gomemcache/memcache  
24 - - go get github.com/gomodule/redigo/redis  
25 - - go get github.com/beego/x2j  
26 - - go get github.com/couchbase/go-couchbase  
27 - - go get github.com/beego/goyaml2  
28 - - go get gopkg.in/yaml.v2  
29 - - go get github.com/belogik/goes  
30 - - go get github.com/siddontang/ledisdb/config  
31 - - go get github.com/siddontang/ledisdb/ledis  
32 - - go get github.com/ssdb/gossdb/ssdb  
33 - - go get github.com/cloudflare/golz4  
34 - - go get github.com/gogo/protobuf/proto  
35 - - go get github.com/Knetic/govaluate  
36 - - go get github.com/casbin/casbin  
37 - - go get -u honnef.co/go/tools/cmd/gosimple  
38 - - go get -u github.com/mdempsky/unconvert  
39 - - go get -u github.com/gordonklaus/ineffassign  
40 - - go get -u github.com/golang/lint/golint  
41 - - go get -u github.com/go-redis/redis  
42 -before_script:  
43 - - psql --version  
44 - - sh -c "if [ '$ORM_DRIVER' = 'postgres' ]; then psql -c 'create database orm_test;' -U postgres; fi"  
45 - - sh -c "if [ '$ORM_DRIVER' = 'mysql' ]; then mysql -u root -e 'create database orm_test;'; fi"  
46 - - sh -c "if [ '$ORM_DRIVER' = 'sqlite' ]; then touch $TRAVIS_BUILD_DIR/orm_test.db; fi"  
47 - - sh -c "if [ $(go version) == *1.[5-9]* ]; then go get github.com/golang/lint/golint; golint ./...; fi"  
48 - - sh -c "if [ $(go version) == *1.[5-9]* ]; then go tool vet .; fi"  
49 - - mkdir -p res/var  
50 - - ./ssdb/ssdb-server ./ssdb/ssdb.conf -d  
51 -after_script:  
52 - -killall -w ssdb-server  
53 - - rm -rf ./res/var/*  
54 -script:  
55 - - go test -v ./...  
56 - - gosimple -ignore "$(cat .gosimpleignore)" $(go list ./... | grep -v /vendor/)  
57 - - unconvert $(go list ./... | grep -v /vendor/)  
58 - - ineffassign .  
59 - - find . ! \( -path './vendor' -prune \) -type f -name '*.go' -print0 | xargs -0 gofmt -l -s  
60 - - golint ./...  
61 -addons:  
62 - postgresql: "9.4"  
1 -# Contributing to beego  
2 -  
3 -beego is an open source project.  
4 -  
5 -It is the work of hundreds of contributors. We appreciate your help!  
6 -  
7 -Here are instructions to get you started. They are probably not perfect,  
8 -please let us know if anything feels wrong or incomplete.  
9 -  
10 -## Contribution guidelines  
11 -  
12 -### Pull requests  
13 -  
14 -First of all. beego follow the gitflow. So please send you pull request  
15 -to **develop** branch. We will close the pull request to master branch.  
16 -  
17 -We are always happy to receive pull requests, and do our best to  
18 -review them as fast as possible. Not sure if that typo is worth a pull  
19 -request? Do it! We will appreciate it.  
20 -  
21 -If your pull request is not accepted on the first try, don't be  
22 -discouraged! Sometimes we can make a mistake, please do more explaining  
23 -for us. We will appreciate it.  
24 -  
25 -We're trying very hard to keep beego simple and fast. We don't want it  
26 -to do everything for everybody. This means that we might decide against  
27 -incorporating a new feature. But we will give you some advice on how to  
28 -do it in other way.  
29 -  
30 -### Create issues  
31 -  
32 -Any significant improvement should be documented as [a GitHub  
33 -issue](https://github.com/astaxie/beego/issues) before anybody  
34 -starts working on it.  
35 -  
36 -Also when filing an issue, make sure to answer these five questions:  
37 -  
38 -- What version of beego are you using (bee version)?  
39 -- What operating system and processor architecture are you using?  
40 -- What did you do?  
41 -- What did you expect to see?  
42 -- What did you see instead?  
43 -  
44 -### but check existing issues and docs first!  
45 -  
46 -Please take a moment to check that an issue doesn't already exist  
47 -documenting your bug report or improvement proposal. If it does, it  
48 -never hurts to add a quick "+1" or "I have this problem too". This will  
49 -help prioritize the most common problems and requests.  
50 -  
51 -Also if you don't know how to use it. please make sure you have read though  
52 -the docs in http://beego.me/docs  
1 -Copyright 2014 astaxie  
2 -  
3 -Licensed under the Apache License, Version 2.0 (the "License");  
4 -you may not use this file except in compliance with the License.  
5 -You may obtain a copy of the License at  
6 -  
7 - http://www.apache.org/licenses/LICENSE-2.0  
8 -  
9 -Unless required by applicable law or agreed to in writing, software  
10 -distributed under the License is distributed on an "AS IS" BASIS,  
11 -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -See the License for the specific language governing permissions and  
13 -limitations under the License.  
1 -# Beego [![Build Status](https://travis-ci.org/astaxie/beego.svg?branch=master)](https://travis-ci.org/astaxie/beego) [![GoDoc](http://godoc.org/github.com/astaxie/beego?status.svg)](http://godoc.org/github.com/astaxie/beego) [![Foundation](https://img.shields.io/badge/Golang-Foundation-green.svg)](http://golangfoundation.org) [![Go Report Card](https://goreportcard.com/badge/github.com/astaxie/beego)](https://goreportcard.com/report/github.com/astaxie/beego)  
2 -  
3 -  
4 -beego is used for rapid development of RESTful APIs, web apps and backend services in Go.  
5 -It is inspired by Tornado, Sinatra and Flask. beego has some Go-specific features such as interfaces and struct embedding.  
6 -  
7 -###### More info at [beego.me](http://beego.me).  
8 -  
9 -## Quick Start  
10 -  
11 -#### Download and install  
12 -  
13 - go get github.com/astaxie/beego  
14 -  
15 -#### Create file `hello.go`  
16 -```go  
17 -package main  
18 -  
19 -import "github.com/astaxie/beego"  
20 -  
21 -func main(){  
22 - beego.Run()  
23 -}  
24 -```  
25 -#### Build and run  
26 -  
27 - go build hello.go  
28 - ./hello  
29 -  
30 -#### Go to [http://localhost:8080](http://localhost:8080)  
31 -  
32 -Congratulations! You've just built your first **beego** app.  
33 -  
34 -###### Please see [Documentation](http://beego.me/docs) for more.  
35 -  
36 -## Features  
37 -  
38 -* RESTful support  
39 -* MVC architecture  
40 -* Modularity  
41 -* Auto API documents  
42 -* Annotation router  
43 -* Namespace  
44 -* Powerful development tools  
45 -* Full stack for Web & API  
46 -  
47 -## Documentation  
48 -  
49 -* [English](http://beego.me/docs/intro/)  
50 -* [中文文档](http://beego.me/docs/intro/)  
51 -* [Русский](http://beego.me/docs/intro/)  
52 -  
53 -## Community  
54 -  
55 -* [http://beego.me/community](http://beego.me/community)  
56 -* Welcome to join us in Slack: [https://beego.slack.com](https://beego.slack.com), you can get invited from [here](https://github.com/beego/beedoc/issues/232)  
57 -  
58 -## License  
59 -  
60 -beego source code is licensed under the Apache Licence, Version 2.0  
61 -(http://www.apache.org/licenses/LICENSE-2.0.html).  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package beego  
16 -  
17 -import (  
18 - "bytes"  
19 - "encoding/json"  
20 - "fmt"  
21 - "net/http"  
22 - "os"  
23 - "text/template"  
24 - "time"  
25 -  
26 - "reflect"  
27 -  
28 - "github.com/astaxie/beego/grace"  
29 - "github.com/astaxie/beego/logs"  
30 - "github.com/astaxie/beego/toolbox"  
31 - "github.com/astaxie/beego/utils"  
32 -)  
33 -  
34 -// BeeAdminApp is the default adminApp used by admin module.  
35 -var beeAdminApp *adminApp  
36 -  
37 -// FilterMonitorFunc is default monitor filter when admin module is enable.  
38 -// if this func returns, admin module records qbs for this request by condition of this function logic.  
39 -// usage:  
40 -// func MyFilterMonitor(method, requestPath string, t time.Duration, pattern string, statusCode int) bool {  
41 -// if method == "POST" {  
42 -// return false  
43 -// }  
44 -// if t.Nanoseconds() < 100 {  
45 -// return false  
46 -// }  
47 -// if strings.HasPrefix(requestPath, "/astaxie") {  
48 -// return false  
49 -// }  
50 -// return true  
51 -// }  
52 -// beego.FilterMonitorFunc = MyFilterMonitor.  
53 -var FilterMonitorFunc func(string, string, time.Duration, string, int) bool  
54 -  
55 -func init() {  
56 - beeAdminApp = &adminApp{  
57 - routers: make(map[string]http.HandlerFunc),  
58 - }  
59 - beeAdminApp.Route("/", adminIndex)  
60 - beeAdminApp.Route("/qps", qpsIndex)  
61 - beeAdminApp.Route("/prof", profIndex)  
62 - beeAdminApp.Route("/healthcheck", healthcheck)  
63 - beeAdminApp.Route("/task", taskStatus)  
64 - beeAdminApp.Route("/listconf", listConf)  
65 - FilterMonitorFunc = func(string, string, time.Duration, string, int) bool { return true }  
66 -}  
67 -  
68 -// AdminIndex is the default http.Handler for admin module.  
69 -// it matches url pattern "/".  
70 -func adminIndex(rw http.ResponseWriter, r *http.Request) {  
71 - execTpl(rw, map[interface{}]interface{}{}, indexTpl, defaultScriptsTpl)  
72 -}  
73 -  
74 -// QpsIndex is the http.Handler for writing qbs statistics map result info in http.ResponseWriter.  
75 -// it's registered with url pattern "/qbs" in admin module.  
76 -func qpsIndex(rw http.ResponseWriter, r *http.Request) {  
77 - data := make(map[interface{}]interface{})  
78 - data["Content"] = toolbox.StatisticsMap.GetMap()  
79 -  
80 - // do html escape before display path, avoid xss  
81 - if content, ok := (data["Content"]).(map[string]interface{}); ok {  
82 - if resultLists, ok := (content["Data"]).([][]string); ok {  
83 - for i := range resultLists {  
84 - if len(resultLists[i]) > 0 {  
85 - resultLists[i][0] = template.HTMLEscapeString(resultLists[i][0])  
86 - }  
87 - }  
88 - }  
89 - }  
90 -  
91 - execTpl(rw, data, qpsTpl, defaultScriptsTpl)  
92 -}  
93 -  
94 -// ListConf is the http.Handler of displaying all beego configuration values as key/value pair.  
95 -// it's registered with url pattern "/listconf" in admin module.  
96 -func listConf(rw http.ResponseWriter, r *http.Request) {  
97 - r.ParseForm()  
98 - command := r.Form.Get("command")  
99 - if command == "" {  
100 - rw.Write([]byte("command not support"))  
101 - return  
102 - }  
103 -  
104 - data := make(map[interface{}]interface{})  
105 - switch command {  
106 - case "conf":  
107 - m := make(map[string]interface{})  
108 - list("BConfig", BConfig, m)  
109 - m["AppConfigPath"] = appConfigPath  
110 - m["AppConfigProvider"] = appConfigProvider  
111 - tmpl := template.Must(template.New("dashboard").Parse(dashboardTpl))  
112 - tmpl = template.Must(tmpl.Parse(configTpl))  
113 - tmpl = template.Must(tmpl.Parse(defaultScriptsTpl))  
114 -  
115 - data["Content"] = m  
116 -  
117 - tmpl.Execute(rw, data)  
118 -  
119 - case "router":  
120 - content := PrintTree()  
121 - content["Fields"] = []string{  
122 - "Router Pattern",  
123 - "Methods",  
124 - "Controller",  
125 - }  
126 - data["Content"] = content  
127 - data["Title"] = "Routers"  
128 - execTpl(rw, data, routerAndFilterTpl, defaultScriptsTpl)  
129 - case "filter":  
130 - var (  
131 - content = map[string]interface{}{  
132 - "Fields": []string{  
133 - "Router Pattern",  
134 - "Filter Function",  
135 - },  
136 - }  
137 - filterTypes = []string{}  
138 - filterTypeData = make(map[string]interface{})  
139 - )  
140 -  
141 - if BeeApp.Handlers.enableFilter {  
142 - var filterType string  
143 - for k, fr := range map[int]string{  
144 - BeforeStatic: "Before Static",  
145 - BeforeRouter: "Before Router",  
146 - BeforeExec: "Before Exec",  
147 - AfterExec: "After Exec",  
148 - FinishRouter: "Finish Router"} {  
149 - if bf := BeeApp.Handlers.filters[k]; len(bf) > 0 {  
150 - filterType = fr  
151 - filterTypes = append(filterTypes, filterType)  
152 - resultList := new([][]string)  
153 - for _, f := range bf {  
154 - var result = []string{  
155 - f.pattern,  
156 - utils.GetFuncName(f.filterFunc),  
157 - }  
158 - *resultList = append(*resultList, result)  
159 - }  
160 - filterTypeData[filterType] = resultList  
161 - }  
162 - }  
163 - }  
164 -  
165 - content["Data"] = filterTypeData  
166 - content["Methods"] = filterTypes  
167 -  
168 - data["Content"] = content  
169 - data["Title"] = "Filters"  
170 - execTpl(rw, data, routerAndFilterTpl, defaultScriptsTpl)  
171 - default:  
172 - rw.Write([]byte("command not support"))  
173 - }  
174 -}  
175 -  
176 -func list(root string, p interface{}, m map[string]interface{}) {  
177 - pt := reflect.TypeOf(p)  
178 - pv := reflect.ValueOf(p)  
179 - if pt.Kind() == reflect.Ptr {  
180 - pt = pt.Elem()  
181 - pv = pv.Elem()  
182 - }  
183 - for i := 0; i < pv.NumField(); i++ {  
184 - var key string  
185 - if root == "" {  
186 - key = pt.Field(i).Name  
187 - } else {  
188 - key = root + "." + pt.Field(i).Name  
189 - }  
190 - if pv.Field(i).Kind() == reflect.Struct {  
191 - list(key, pv.Field(i).Interface(), m)  
192 - } else {  
193 - m[key] = pv.Field(i).Interface()  
194 - }  
195 - }  
196 -}  
197 -  
198 -// PrintTree prints all registered routers.  
199 -func PrintTree() map[string]interface{} {  
200 - var (  
201 - content = map[string]interface{}{}  
202 - methods = []string{}  
203 - methodsData = make(map[string]interface{})  
204 - )  
205 - for method, t := range BeeApp.Handlers.routers {  
206 -  
207 - resultList := new([][]string)  
208 -  
209 - printTree(resultList, t)  
210 -  
211 - methods = append(methods, method)  
212 - methodsData[method] = resultList  
213 - }  
214 -  
215 - content["Data"] = methodsData  
216 - content["Methods"] = methods  
217 - return content  
218 -}  
219 -  
220 -func printTree(resultList *[][]string, t *Tree) {  
221 - for _, tr := range t.fixrouters {  
222 - printTree(resultList, tr)  
223 - }  
224 - if t.wildcard != nil {  
225 - printTree(resultList, t.wildcard)  
226 - }  
227 - for _, l := range t.leaves {  
228 - if v, ok := l.runObject.(*ControllerInfo); ok {  
229 - if v.routerType == routerTypeBeego {  
230 - var result = []string{  
231 - v.pattern,  
232 - fmt.Sprintf("%s", v.methods),  
233 - v.controllerType.String(),  
234 - }  
235 - *resultList = append(*resultList, result)  
236 - } else if v.routerType == routerTypeRESTFul {  
237 - var result = []string{  
238 - v.pattern,  
239 - fmt.Sprintf("%s", v.methods),  
240 - "",  
241 - }  
242 - *resultList = append(*resultList, result)  
243 - } else if v.routerType == routerTypeHandler {  
244 - var result = []string{  
245 - v.pattern,  
246 - "",  
247 - "",  
248 - }  
249 - *resultList = append(*resultList, result)  
250 - }  
251 - }  
252 - }  
253 -}  
254 -  
255 -// ProfIndex is a http.Handler for showing profile command.  
256 -// it's in url pattern "/prof" in admin module.  
257 -func profIndex(rw http.ResponseWriter, r *http.Request) {  
258 - r.ParseForm()  
259 - command := r.Form.Get("command")  
260 - if command == "" {  
261 - return  
262 - }  
263 -  
264 - var (  
265 - format = r.Form.Get("format")  
266 - data = make(map[interface{}]interface{})  
267 - result bytes.Buffer  
268 - )  
269 - toolbox.ProcessInput(command, &result)  
270 - data["Content"] = result.String()  
271 -  
272 - if format == "json" && command == "gc summary" {  
273 - dataJSON, err := json.Marshal(data)  
274 - if err != nil {  
275 - http.Error(rw, err.Error(), http.StatusInternalServerError)  
276 - return  
277 - }  
278 -  
279 - rw.Header().Set("Content-Type", "application/json")  
280 - rw.Write(dataJSON)  
281 - return  
282 - }  
283 -  
284 - data["Title"] = command  
285 - defaultTpl := defaultScriptsTpl  
286 - if command == "gc summary" {  
287 - defaultTpl = gcAjaxTpl  
288 - }  
289 - execTpl(rw, data, profillingTpl, defaultTpl)  
290 -}  
291 -  
292 -// Healthcheck is a http.Handler calling health checking and showing the result.  
293 -// it's in "/healthcheck" pattern in admin module.  
294 -func healthcheck(rw http.ResponseWriter, req *http.Request) {  
295 - var (  
296 - result []string  
297 - data = make(map[interface{}]interface{})  
298 - resultList = new([][]string)  
299 - content = map[string]interface{}{  
300 - "Fields": []string{"Name", "Message", "Status"},  
301 - }  
302 - )  
303 -  
304 - for name, h := range toolbox.AdminCheckList {  
305 - if err := h.Check(); err != nil {  
306 - result = []string{  
307 - "error",  
308 - name,  
309 - err.Error(),  
310 - }  
311 - } else {  
312 - result = []string{  
313 - "success",  
314 - name,  
315 - "OK",  
316 - }  
317 - }  
318 - *resultList = append(*resultList, result)  
319 - }  
320 -  
321 - content["Data"] = resultList  
322 - data["Content"] = content  
323 - data["Title"] = "Health Check"  
324 - execTpl(rw, data, healthCheckTpl, defaultScriptsTpl)  
325 -}  
326 -  
327 -// TaskStatus is a http.Handler with running task status (task name, status and the last execution).  
328 -// it's in "/task" pattern in admin module.  
329 -func taskStatus(rw http.ResponseWriter, req *http.Request) {  
330 - data := make(map[interface{}]interface{})  
331 -  
332 - // Run Task  
333 - req.ParseForm()  
334 - taskname := req.Form.Get("taskname")  
335 - if taskname != "" {  
336 - if t, ok := toolbox.AdminTaskList[taskname]; ok {  
337 - if err := t.Run(); err != nil {  
338 - data["Message"] = []string{"error", fmt.Sprintf("%s", err)}  
339 - }  
340 - data["Message"] = []string{"success", fmt.Sprintf("%s run success,Now the Status is <br>%s", taskname, t.GetStatus())}  
341 - } else {  
342 - data["Message"] = []string{"warning", fmt.Sprintf("there's no task which named: %s", taskname)}  
343 - }  
344 - }  
345 -  
346 - // List Tasks  
347 - content := make(map[string]interface{})  
348 - resultList := new([][]string)  
349 - var fields = []string{  
350 - "Task Name",  
351 - "Task Spec",  
352 - "Task Status",  
353 - "Last Time",  
354 - "",  
355 - }  
356 - for tname, tk := range toolbox.AdminTaskList {  
357 - result := []string{  
358 - tname,  
359 - tk.GetSpec(),  
360 - tk.GetStatus(),  
361 - tk.GetPrev().String(),  
362 - }  
363 - *resultList = append(*resultList, result)  
364 - }  
365 -  
366 - content["Fields"] = fields  
367 - content["Data"] = resultList  
368 - data["Content"] = content  
369 - data["Title"] = "Tasks"  
370 - execTpl(rw, data, tasksTpl, defaultScriptsTpl)  
371 -}  
372 -  
373 -func execTpl(rw http.ResponseWriter, data map[interface{}]interface{}, tpls ...string) {  
374 - tmpl := template.Must(template.New("dashboard").Parse(dashboardTpl))  
375 - for _, tpl := range tpls {  
376 - tmpl = template.Must(tmpl.Parse(tpl))  
377 - }  
378 - tmpl.Execute(rw, data)  
379 -}  
380 -  
381 -// adminApp is an http.HandlerFunc map used as beeAdminApp.  
382 -type adminApp struct {  
383 - routers map[string]http.HandlerFunc  
384 -}  
385 -  
386 -// Route adds http.HandlerFunc to adminApp with url pattern.  
387 -func (admin *adminApp) Route(pattern string, f http.HandlerFunc) {  
388 - admin.routers[pattern] = f  
389 -}  
390 -  
391 -// Run adminApp http server.  
392 -// Its addr is defined in configuration file as adminhttpaddr and adminhttpport.  
393 -func (admin *adminApp) Run() {  
394 - if len(toolbox.AdminTaskList) > 0 {  
395 - toolbox.StartTask()  
396 - }  
397 - addr := BConfig.Listen.AdminAddr  
398 -  
399 - if BConfig.Listen.AdminPort != 0 {  
400 - addr = fmt.Sprintf("%s:%d", BConfig.Listen.AdminAddr, BConfig.Listen.AdminPort)  
401 - }  
402 - for p, f := range admin.routers {  
403 - http.Handle(p, f)  
404 - }  
405 - logs.Info("Admin server Running on %s", addr)  
406 -  
407 - var err error  
408 - if BConfig.Listen.Graceful {  
409 - err = grace.ListenAndServe(addr, nil)  
410 - } else {  
411 - err = http.ListenAndServe(addr, nil)  
412 - }  
413 - if err != nil {  
414 - logs.Critical("Admin ListenAndServe: ", err, fmt.Sprintf("%d", os.Getpid()))  
415 - }  
416 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package beego  
16 -  
17 -var indexTpl = `  
18 -{{define "content"}}  
19 -<h1>Beego Admin Dashboard</h1>  
20 -<p>  
21 -For detail usage please check our document:  
22 -</p>  
23 -<p>  
24 -<a target="_blank" href="http://beego.me/docs/module/toolbox.md">Toolbox</a>  
25 -</p>  
26 -<p>  
27 -<a target="_blank" href="http://beego.me/docs/advantage/monitor.md">Live Monitor</a>  
28 -</p>  
29 -{{.Content}}  
30 -{{end}}`  
31 -  
32 -var profillingTpl = `  
33 -{{define "content"}}  
34 -<h1>{{.Title}}</h1>  
35 -<pre id="content">  
36 -<div>{{.Content}}</div>  
37 -</pre>  
38 -{{end}}`  
39 -  
40 -var defaultScriptsTpl = ``  
41 -  
42 -var gcAjaxTpl = `  
43 -{{define "scripts"}}  
44 -<script type="text/javascript">  
45 - var app = app || {};  
46 -(function() {  
47 - app.$el = $('#content');  
48 - app.getGc = function() {  
49 - var that = this;  
50 - $.ajax("/prof?command=gc%20summary&format=json").done(function(data) {  
51 - that.$el.append($('<p>' + data.Content + '</p>'));  
52 - });  
53 - };  
54 - $(document).ready(function() {  
55 - setInterval(function() {  
56 - app.getGc();  
57 - }, 3000);  
58 - });  
59 -})();  
60 -</script>  
61 -{{end}}  
62 -`  
63 -  
64 -var qpsTpl = `{{define "content"}}  
65 -<h1>Requests statistics</h1>  
66 -<table class="table table-striped table-hover ">  
67 - <thead>  
68 - <tr>  
69 - {{range .Content.Fields}}  
70 - <th>  
71 - {{.}}  
72 - </th>  
73 - {{end}}  
74 - </tr>  
75 - </thead>  
76 -  
77 - <tbody>  
78 - {{range $i, $elem := .Content.Data}}  
79 -  
80 - <tr>  
81 - <td>{{index $elem 0}}</td>  
82 - <td>{{index $elem 1}}</td>  
83 - <td>{{index $elem 2}}</td>  
84 - <td data-order="{{index $elem 3}}">{{index $elem 4}}</td>  
85 - <td data-order="{{index $elem 5}}">{{index $elem 6}}</td>  
86 - <td data-order="{{index $elem 7}}">{{index $elem 8}}</td>  
87 - <td data-order="{{index $elem 9}}">{{index $elem 10}}</td>  
88 - </tr>  
89 - {{end}}  
90 - </tbody>  
91 -  
92 -</table>  
93 -{{end}}`  
94 -  
95 -var configTpl = `  
96 -{{define "content"}}  
97 -<h1>Configurations</h1>  
98 -<pre>  
99 -{{range $index, $elem := .Content}}  
100 -{{$index}}={{$elem}}  
101 -{{end}}  
102 -</pre>  
103 -{{end}}  
104 -`  
105 -  
106 -var routerAndFilterTpl = `{{define "content"}}  
107 -  
108 -  
109 -<h1>{{.Title}}</h1>  
110 -  
111 -{{range .Content.Methods}}  
112 -  
113 -<div class="panel panel-default">  
114 -<div class="panel-heading lead success"><strong>{{.}}</strong></div>  
115 -<div class="panel-body">  
116 -<table class="table table-striped table-hover ">  
117 - <thead>  
118 - <tr>  
119 - {{range $.Content.Fields}}  
120 - <th>  
121 - {{.}}  
122 - </th>  
123 - {{end}}  
124 - </tr>  
125 - </thead>  
126 -  
127 - <tbody>  
128 - {{$slice := index $.Content.Data .}}  
129 - {{range $i, $elem := $slice}}  
130 -  
131 - <tr>  
132 - {{range $elem}}  
133 - <td>  
134 - {{.}}  
135 - </td>  
136 - {{end}}  
137 - </tr>  
138 -  
139 - {{end}}  
140 - </tbody>  
141 -  
142 -</table>  
143 -</div>  
144 -</div>  
145 -{{end}}  
146 -  
147 -  
148 -{{end}}`  
149 -  
150 -var tasksTpl = `{{define "content"}}  
151 -  
152 -<h1>{{.Title}}</h1>  
153 -  
154 -{{if .Message }}  
155 -{{ $messageType := index .Message 0}}  
156 -<p class="message  
157 -{{if eq "error" $messageType}}  
158 -bg-danger  
159 -{{else if eq "success" $messageType}}  
160 -bg-success  
161 -{{else}}  
162 -bg-warning  
163 -{{end}}  
164 -">  
165 -{{index .Message 1}}  
166 -</p>  
167 -{{end}}  
168 -  
169 -  
170 -<table class="table table-striped table-hover ">  
171 -<thead>  
172 -<tr>  
173 -{{range .Content.Fields}}  
174 -<th>  
175 -{{.}}  
176 -</th>  
177 -{{end}}  
178 -</tr>  
179 -</thead>  
180 -  
181 -<tbody>  
182 -{{range $i, $slice := .Content.Data}}  
183 -<tr>  
184 - {{range $slice}}  
185 - <td>  
186 - {{.}}  
187 - </td>  
188 - {{end}}  
189 - <td>  
190 - <a class="btn btn-primary btn-sm" href="/task?taskname={{index $slice 0}}">Run</a>  
191 - </td>  
192 -</tr>  
193 -{{end}}  
194 -</tbody>  
195 -</table>  
196 -  
197 -{{end}}`  
198 -  
199 -var healthCheckTpl = `  
200 -{{define "content"}}  
201 -  
202 -<h1>{{.Title}}</h1>  
203 -<table class="table table-striped table-hover ">  
204 -<thead>  
205 -<tr>  
206 -{{range .Content.Fields}}  
207 - <th>  
208 - {{.}}  
209 - </th>  
210 -{{end}}  
211 -</tr>  
212 -</thead>  
213 -<tbody>  
214 -{{range $i, $slice := .Content.Data}}  
215 - {{ $header := index $slice 0}}  
216 - {{ if eq "success" $header}}  
217 - <tr class="success">  
218 - {{else if eq "error" $header}}  
219 - <tr class="danger">  
220 - {{else}}  
221 - <tr>  
222 - {{end}}  
223 - {{range $j, $elem := $slice}}  
224 - {{if ne $j 0}}  
225 - <td>  
226 - {{$elem}}  
227 - </td>  
228 - {{end}}  
229 - {{end}}  
230 - <td>  
231 - {{$header}}  
232 - </td>  
233 - </tr>  
234 -{{end}}  
235 -  
236 -</tbody>  
237 -</table>  
238 -{{end}}`  
239 -  
240 -// The base dashboardTpl  
241 -var dashboardTpl = `  
242 -<!DOCTYPE html>  
243 -<html lang="en">  
244 -<head>  
245 -<!-- Meta, title, CSS, favicons, etc. -->  
246 -<meta charset="utf-8">  
247 -<meta http-equiv="X-UA-Compatible" content="IE=edge">  
248 -<meta name="viewport" content="width=device-width, initial-scale=1">  
249 -  
250 -<title>  
251 -  
252 -Welcome to Beego Admin Dashboard  
253 -  
254 -</title>  
255 -  
256 -<link href="//maxcdn.bootstrapcdn.com/bootstrap/3.2.0/css/bootstrap.min.css" rel="stylesheet">  
257 -<link href="//cdn.datatables.net/plug-ins/725b2a2115b/integration/bootstrap/3/dataTables.bootstrap.css" rel="stylesheet">  
258 -  
259 -<style type="text/css">  
260 -ul.nav li.dropdown:hover > ul.dropdown-menu {  
261 - display: block;  
262 -}  
263 -#logo {  
264 - width: 102px;  
265 - height: 32px;  
266 - margin-top: 5px;  
267 -}  
268 -.message {  
269 - padding: 15px;  
270 -}  
271 -</style>  
272 -  
273 -</head>  
274 -<body>  
275 -  
276 -<header class="navbar navbar-default navbar-static-top bs-docs-nav" id="top" role="banner">  
277 -<div class="container">  
278 -<div class="navbar-header">  
279 -<button class="navbar-toggle" type="button" data-toggle="collapse" data-target=".bs-navbar-collapse">  
280 -<span class="sr-only">Toggle navigation</span>  
281 -<span class="icon-bar"></span>  
282 -<span class="icon-bar"></span>  
283 -<span class="icon-bar"></span>  
284 -</button>  
285 -  
286 -<a href="/">  
287 -<img id="logo" src=""/>  
288 -</a>  
289 -  
290 -</div>  
291 -<nav class="collapse navbar-collapse bs-navbar-collapse" role="navigation">  
292 -<ul class="nav navbar-nav">  
293 -<li>  
294 -<a href="/qps">  
295 -Requests statistics  
296 -</a>  
297 -</li>  
298 -<li>  
299 -  
300 -<li class="dropdown">  
301 -<a href="#" class="dropdown-toggle disabled" data-toggle="dropdown">Performance profiling<span class="caret"></span></a>  
302 -<ul class="dropdown-menu" role="menu">  
303 -  
304 -<li><a href="/prof?command=lookup goroutine">lookup goroutine</a></li>  
305 -<li><a href="/prof?command=lookup heap">lookup heap</a></li>  
306 -<li><a href="/prof?command=lookup threadcreate">lookup threadcreate</a></li>  
307 -<li><a href="/prof?command=lookup block">lookup block</a></li>  
308 -<li><a href="/prof?command=get cpuprof">get cpuprof</a></li>  
309 -<li><a href="/prof?command=get memprof">get memprof</a></li>  
310 -<li><a href="/prof?command=gc summary">gc summary</a></li>  
311 -  
312 -</ul>  
313 -</li>  
314 -  
315 -<li>  
316 -<a href="/healthcheck">  
317 -Healthcheck  
318 -</a>  
319 -</li>  
320 -  
321 -<li>  
322 -<a href="/task" class="dropdown-toggle disabled" data-toggle="dropdown">Tasks</a>  
323 -</li>  
324 -  
325 -<li class="dropdown">  
326 -<a href="#" class="dropdown-toggle disabled" data-toggle="dropdown">Config Status<span class="caret"></span></a>  
327 -<ul class="dropdown-menu" role="menu">  
328 -<li><a href="/listconf?command=conf">Configs</a></li>  
329 -<li><a href="/listconf?command=router">Routers</a></li>  
330 -<li><a href="/listconf?command=filter">Filters</a></li>  
331 -</ul>  
332 -</li>  
333 -</ul>  
334 -</nav>  
335 -</div>  
336 -</header>  
337 -  
338 -<div class="container">  
339 -{{template "content" .}}  
340 -</div>  
341 -  
342 -<script src="//code.jquery.com/jquery-1.11.1.min.js"></script>  
343 -<script src="//maxcdn.bootstrapcdn.com/bootstrap/3.2.0/js/bootstrap.min.js"></script>  
344 -<script src="//cdn.datatables.net/1.10.2/js/jquery.dataTables.min.js"></script>  
345 -<script src="//cdn.datatables.net/plug-ins/725b2a2115b/integration/bootstrap/3/dataTables.bootstrap.js  
346 -"></script>  
347 -  
348 -<script type="text/javascript">  
349 -$(document).ready(function() {  
350 - $('.table').dataTable();  
351 -});  
352 -</script>  
353 -{{template "scripts" .}}  
354 -</body>  
355 -</html>  
356 -`  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package beego  
16 -  
17 -import (  
18 - "crypto/tls"  
19 - "crypto/x509"  
20 - "fmt"  
21 - "io/ioutil"  
22 - "net"  
23 - "net/http"  
24 - "net/http/fcgi"  
25 - "os"  
26 - "path"  
27 - "strings"  
28 - "time"  
29 -  
30 - "github.com/astaxie/beego/grace"  
31 - "github.com/astaxie/beego/logs"  
32 - "github.com/astaxie/beego/utils"  
33 - "golang.org/x/crypto/acme/autocert"  
34 -)  
35 -  
36 -var (  
37 - // BeeApp is an application instance  
38 - BeeApp *App  
39 -)  
40 -  
41 -func init() {  
42 - // create beego application  
43 - BeeApp = NewApp()  
44 -}  
45 -  
46 -// App defines beego application with a new PatternServeMux.  
47 -type App struct {  
48 - Handlers *ControllerRegister  
49 - Server *http.Server  
50 -}  
51 -  
52 -// NewApp returns a new beego application.  
53 -func NewApp() *App {  
54 - cr := NewControllerRegister()  
55 - app := &App{Handlers: cr, Server: &http.Server{}}  
56 - return app  
57 -}  
58 -  
59 -// MiddleWare function for http.Handler  
60 -type MiddleWare func(http.Handler) http.Handler  
61 -  
62 -// Run beego application.  
63 -func (app *App) Run(mws ...MiddleWare) {  
64 - addr := BConfig.Listen.HTTPAddr  
65 -  
66 - if BConfig.Listen.HTTPPort != 0 {  
67 - addr = fmt.Sprintf("%s:%d", BConfig.Listen.HTTPAddr, BConfig.Listen.HTTPPort)  
68 - }  
69 -  
70 - var (  
71 - err error  
72 - l net.Listener  
73 - endRunning = make(chan bool, 1)  
74 - )  
75 -  
76 - // run cgi server  
77 - if BConfig.Listen.EnableFcgi {  
78 - if BConfig.Listen.EnableStdIo {  
79 - if err = fcgi.Serve(nil, app.Handlers); err == nil { // standard I/O  
80 - logs.Info("Use FCGI via standard I/O")  
81 - } else {  
82 - logs.Critical("Cannot use FCGI via standard I/O", err)  
83 - }  
84 - return  
85 - }  
86 - if BConfig.Listen.HTTPPort == 0 {  
87 - // remove the Socket file before start  
88 - if utils.FileExists(addr) {  
89 - os.Remove(addr)  
90 - }  
91 - l, err = net.Listen("unix", addr)  
92 - } else {  
93 - l, err = net.Listen("tcp", addr)  
94 - }  
95 - if err != nil {  
96 - logs.Critical("Listen: ", err)  
97 - }  
98 - if err = fcgi.Serve(l, app.Handlers); err != nil {  
99 - logs.Critical("fcgi.Serve: ", err)  
100 - }  
101 - return  
102 - }  
103 -  
104 - app.Server.Handler = app.Handlers  
105 - for i := len(mws) - 1; i >= 0; i-- {  
106 - if mws[i] == nil {  
107 - continue  
108 - }  
109 - app.Server.Handler = mws[i](app.Server.Handler)  
110 - }  
111 - app.Server.ReadTimeout = time.Duration(BConfig.Listen.ServerTimeOut) * time.Second  
112 - app.Server.WriteTimeout = time.Duration(BConfig.Listen.ServerTimeOut) * time.Second  
113 - app.Server.ErrorLog = logs.GetLogger("HTTP")  
114 -  
115 - // run graceful mode  
116 - if BConfig.Listen.Graceful {  
117 - httpsAddr := BConfig.Listen.HTTPSAddr  
118 - app.Server.Addr = httpsAddr  
119 - if BConfig.Listen.EnableHTTPS || BConfig.Listen.EnableMutualHTTPS {  
120 - go func() {  
121 - time.Sleep(1000 * time.Microsecond)  
122 - if BConfig.Listen.HTTPSPort != 0 {  
123 - httpsAddr = fmt.Sprintf("%s:%d", BConfig.Listen.HTTPSAddr, BConfig.Listen.HTTPSPort)  
124 - app.Server.Addr = httpsAddr  
125 - }  
126 - server := grace.NewServer(httpsAddr, app.Handlers)  
127 - server.Server.ReadTimeout = app.Server.ReadTimeout  
128 - server.Server.WriteTimeout = app.Server.WriteTimeout  
129 - if BConfig.Listen.EnableMutualHTTPS {  
130 - if err := server.ListenAndServeMutualTLS(BConfig.Listen.HTTPSCertFile, BConfig.Listen.HTTPSKeyFile, BConfig.Listen.TrustCaFile); err != nil {  
131 - logs.Critical("ListenAndServeTLS: ", err, fmt.Sprintf("%d", os.Getpid()))  
132 - time.Sleep(100 * time.Microsecond)  
133 - endRunning <- true  
134 - }  
135 - } else {  
136 - if BConfig.Listen.AutoTLS {  
137 - m := autocert.Manager{  
138 - Prompt: autocert.AcceptTOS,  
139 - HostPolicy: autocert.HostWhitelist(BConfig.Listen.Domains...),  
140 - Cache: autocert.DirCache(BConfig.Listen.TLSCacheDir),  
141 - }  
142 - app.Server.TLSConfig = &tls.Config{GetCertificate: m.GetCertificate}  
143 - BConfig.Listen.HTTPSCertFile, BConfig.Listen.HTTPSKeyFile = "", ""  
144 - }  
145 - if err := server.ListenAndServeTLS(BConfig.Listen.HTTPSCertFile, BConfig.Listen.HTTPSKeyFile); err != nil {  
146 - logs.Critical("ListenAndServeTLS: ", err, fmt.Sprintf("%d", os.Getpid()))  
147 - time.Sleep(100 * time.Microsecond)  
148 - endRunning <- true  
149 - }  
150 - }  
151 - }()  
152 - }  
153 - if BConfig.Listen.EnableHTTP {  
154 - go func() {  
155 - server := grace.NewServer(addr, app.Handlers)  
156 - server.Server.ReadTimeout = app.Server.ReadTimeout  
157 - server.Server.WriteTimeout = app.Server.WriteTimeout  
158 - if BConfig.Listen.ListenTCP4 {  
159 - server.Network = "tcp4"  
160 - }  
161 - if err := server.ListenAndServe(); err != nil {  
162 - logs.Critical("ListenAndServe: ", err, fmt.Sprintf("%d", os.Getpid()))  
163 - time.Sleep(100 * time.Microsecond)  
164 - endRunning <- true  
165 - }  
166 - }()  
167 - }  
168 - <-endRunning  
169 - return  
170 - }  
171 -  
172 - // run normal mode  
173 - if BConfig.Listen.EnableHTTPS || BConfig.Listen.EnableMutualHTTPS {  
174 - go func() {  
175 - time.Sleep(1000 * time.Microsecond)  
176 - if BConfig.Listen.HTTPSPort != 0 {  
177 - app.Server.Addr = fmt.Sprintf("%s:%d", BConfig.Listen.HTTPSAddr, BConfig.Listen.HTTPSPort)  
178 - } else if BConfig.Listen.EnableHTTP {  
179 - BeeLogger.Info("Start https server error, conflict with http. Please reset https port")  
180 - return  
181 - }  
182 - logs.Info("https server Running on https://%s", app.Server.Addr)  
183 - if BConfig.Listen.AutoTLS {  
184 - m := autocert.Manager{  
185 - Prompt: autocert.AcceptTOS,  
186 - HostPolicy: autocert.HostWhitelist(BConfig.Listen.Domains...),  
187 - Cache: autocert.DirCache(BConfig.Listen.TLSCacheDir),  
188 - }  
189 - app.Server.TLSConfig = &tls.Config{GetCertificate: m.GetCertificate}  
190 - BConfig.Listen.HTTPSCertFile, BConfig.Listen.HTTPSKeyFile = "", ""  
191 - } else if BConfig.Listen.EnableMutualHTTPS {  
192 - pool := x509.NewCertPool()  
193 - data, err := ioutil.ReadFile(BConfig.Listen.TrustCaFile)  
194 - if err != nil {  
195 - BeeLogger.Info("MutualHTTPS should provide TrustCaFile")  
196 - return  
197 - }  
198 - pool.AppendCertsFromPEM(data)  
199 - app.Server.TLSConfig = &tls.Config{  
200 - ClientCAs: pool,  
201 - ClientAuth: tls.RequireAndVerifyClientCert,  
202 - }  
203 - }  
204 - if err := app.Server.ListenAndServeTLS(BConfig.Listen.HTTPSCertFile, BConfig.Listen.HTTPSKeyFile); err != nil {  
205 - logs.Critical("ListenAndServeTLS: ", err)  
206 - time.Sleep(100 * time.Microsecond)  
207 - endRunning <- true  
208 - }  
209 - }()  
210 -  
211 - }  
212 - if BConfig.Listen.EnableHTTP {  
213 - go func() {  
214 - app.Server.Addr = addr  
215 - logs.Info("http server Running on http://%s", app.Server.Addr)  
216 - if BConfig.Listen.ListenTCP4 {  
217 - ln, err := net.Listen("tcp4", app.Server.Addr)  
218 - if err != nil {  
219 - logs.Critical("ListenAndServe: ", err)  
220 - time.Sleep(100 * time.Microsecond)  
221 - endRunning <- true  
222 - return  
223 - }  
224 - if err = app.Server.Serve(ln); err != nil {  
225 - logs.Critical("ListenAndServe: ", err)  
226 - time.Sleep(100 * time.Microsecond)  
227 - endRunning <- true  
228 - return  
229 - }  
230 - } else {  
231 - if err := app.Server.ListenAndServe(); err != nil {  
232 - logs.Critical("ListenAndServe: ", err)  
233 - time.Sleep(100 * time.Microsecond)  
234 - endRunning <- true  
235 - }  
236 - }  
237 - }()  
238 - }  
239 - <-endRunning  
240 -}  
241 -  
242 -// Router adds a patterned controller handler to BeeApp.  
243 -// it's an alias method of App.Router.  
244 -// usage:  
245 -// simple router  
246 -// beego.Router("/admin", &admin.UserController{})  
247 -// beego.Router("/admin/index", &admin.ArticleController{})  
248 -//  
249 -// regex router  
250 -//  
251 -// beego.Router("/api/:id([0-9]+)", &controllers.RController{})  
252 -//  
253 -// custom rules  
254 -// beego.Router("/api/list",&RestController{},"*:ListFood")  
255 -// beego.Router("/api/create",&RestController{},"post:CreateFood")  
256 -// beego.Router("/api/update",&RestController{},"put:UpdateFood")  
257 -// beego.Router("/api/delete",&RestController{},"delete:DeleteFood")  
258 -func Router(rootpath string, c ControllerInterface, mappingMethods ...string) *App {  
259 - BeeApp.Handlers.Add(rootpath, c, mappingMethods...)  
260 - return BeeApp  
261 -}  
262 -  
263 -// UnregisterFixedRoute unregisters the route with the specified fixedRoute. It is particularly useful  
264 -// in web applications that inherit most routes from a base webapp via the underscore  
265 -// import, and aim to overwrite only certain paths.  
266 -// The method parameter can be empty or "*" for all HTTP methods, or a particular  
267 -// method type (e.g. "GET" or "POST") for selective removal.  
268 -//  
269 -// Usage (replace "GET" with "*" for all methods):  
270 -// beego.UnregisterFixedRoute("/yourpreviouspath", "GET")  
271 -// beego.Router("/yourpreviouspath", yourControllerAddress, "get:GetNewPage")  
272 -func UnregisterFixedRoute(fixedRoute string, method string) *App {  
273 - subPaths := splitPath(fixedRoute)  
274 - if method == "" || method == "*" {  
275 - for m := range HTTPMETHOD {  
276 - if _, ok := BeeApp.Handlers.routers[m]; !ok {  
277 - continue  
278 - }  
279 - if BeeApp.Handlers.routers[m].prefix == strings.Trim(fixedRoute, "/ ") {  
280 - findAndRemoveSingleTree(BeeApp.Handlers.routers[m])  
281 - continue  
282 - }  
283 - findAndRemoveTree(subPaths, BeeApp.Handlers.routers[m], m)  
284 - }  
285 - return BeeApp  
286 - }  
287 - // Single HTTP method  
288 - um := strings.ToUpper(method)  
289 - if _, ok := BeeApp.Handlers.routers[um]; ok {  
290 - if BeeApp.Handlers.routers[um].prefix == strings.Trim(fixedRoute, "/ ") {  
291 - findAndRemoveSingleTree(BeeApp.Handlers.routers[um])  
292 - return BeeApp  
293 - }  
294 - findAndRemoveTree(subPaths, BeeApp.Handlers.routers[um], um)  
295 - }  
296 - return BeeApp  
297 -}  
298 -  
299 -func findAndRemoveTree(paths []string, entryPointTree *Tree, method string) {  
300 - for i := range entryPointTree.fixrouters {  
301 - if entryPointTree.fixrouters[i].prefix == paths[0] {  
302 - if len(paths) == 1 {  
303 - if len(entryPointTree.fixrouters[i].fixrouters) > 0 {  
304 - // If the route had children subtrees, remove just the functional leaf,  
305 - // to allow children to function as before  
306 - if len(entryPointTree.fixrouters[i].leaves) > 0 {  
307 - entryPointTree.fixrouters[i].leaves[0] = nil  
308 - entryPointTree.fixrouters[i].leaves = entryPointTree.fixrouters[i].leaves[1:]  
309 - }  
310 - } else {  
311 - // Remove the *Tree from the fixrouters slice  
312 - entryPointTree.fixrouters[i] = nil  
313 -  
314 - if i == len(entryPointTree.fixrouters)-1 {  
315 - entryPointTree.fixrouters = entryPointTree.fixrouters[:i]  
316 - } else {  
317 - entryPointTree.fixrouters = append(entryPointTree.fixrouters[:i], entryPointTree.fixrouters[i+1:len(entryPointTree.fixrouters)]...)  
318 - }  
319 - }  
320 - return  
321 - }  
322 - findAndRemoveTree(paths[1:], entryPointTree.fixrouters[i], method)  
323 - }  
324 - }  
325 -}  
326 -  
327 -func findAndRemoveSingleTree(entryPointTree *Tree) {  
328 - if entryPointTree == nil {  
329 - return  
330 - }  
331 - if len(entryPointTree.fixrouters) > 0 {  
332 - // If the route had children subtrees, remove just the functional leaf,  
333 - // to allow children to function as before  
334 - if len(entryPointTree.leaves) > 0 {  
335 - entryPointTree.leaves[0] = nil  
336 - entryPointTree.leaves = entryPointTree.leaves[1:]  
337 - }  
338 - }  
339 -}  
340 -  
341 -// Include will generate router file in the router/xxx.go from the controller's comments  
342 -// usage:  
343 -// beego.Include(&BankAccount{}, &OrderController{},&RefundController{},&ReceiptController{})  
344 -// type BankAccount struct{  
345 -// beego.Controller  
346 -// }  
347 -//  
348 -// register the function  
349 -// func (b *BankAccount)Mapping(){  
350 -// b.Mapping("ShowAccount" , b.ShowAccount)  
351 -// b.Mapping("ModifyAccount", b.ModifyAccount)  
352 -//}  
353 -//  
354 -// //@router /account/:id [get]  
355 -// func (b *BankAccount) ShowAccount(){  
356 -// //logic  
357 -// }  
358 -//  
359 -//  
360 -// //@router /account/:id [post]  
361 -// func (b *BankAccount) ModifyAccount(){  
362 -// //logic  
363 -// }  
364 -//  
365 -// the comments @router url methodlist  
366 -// url support all the function Router's pattern  
367 -// methodlist [get post head put delete options *]  
368 -func Include(cList ...ControllerInterface) *App {  
369 - BeeApp.Handlers.Include(cList...)  
370 - return BeeApp  
371 -}  
372 -  
373 -// RESTRouter adds a restful controller handler to BeeApp.  
374 -// its' controller implements beego.ControllerInterface and  
375 -// defines a param "pattern/:objectId" to visit each resource.  
376 -func RESTRouter(rootpath string, c ControllerInterface) *App {  
377 - Router(rootpath, c)  
378 - Router(path.Join(rootpath, ":objectId"), c)  
379 - return BeeApp  
380 -}  
381 -  
382 -// AutoRouter adds defined controller handler to BeeApp.  
383 -// it's same to App.AutoRouter.  
384 -// if beego.AddAuto(&MainContorlller{}) and MainController has methods List and Page,  
385 -// visit the url /main/list to exec List function or /main/page to exec Page function.  
386 -func AutoRouter(c ControllerInterface) *App {  
387 - BeeApp.Handlers.AddAuto(c)  
388 - return BeeApp  
389 -}  
390 -  
391 -// AutoPrefix adds controller handler to BeeApp with prefix.  
392 -// it's same to App.AutoRouterWithPrefix.  
393 -// if beego.AutoPrefix("/admin",&MainContorlller{}) and MainController has methods List and Page,  
394 -// visit the url /admin/main/list to exec List function or /admin/main/page to exec Page function.  
395 -func AutoPrefix(prefix string, c ControllerInterface) *App {  
396 - BeeApp.Handlers.AddAutoPrefix(prefix, c)  
397 - return BeeApp  
398 -}  
399 -  
400 -// Get used to register router for Get method  
401 -// usage:  
402 -// beego.Get("/", func(ctx *context.Context){  
403 -// ctx.Output.Body("hello world")  
404 -// })  
405 -func Get(rootpath string, f FilterFunc) *App {  
406 - BeeApp.Handlers.Get(rootpath, f)  
407 - return BeeApp  
408 -}  
409 -  
410 -// Post used to register router for Post method  
411 -// usage:  
412 -// beego.Post("/api", func(ctx *context.Context){  
413 -// ctx.Output.Body("hello world")  
414 -// })  
415 -func Post(rootpath string, f FilterFunc) *App {  
416 - BeeApp.Handlers.Post(rootpath, f)  
417 - return BeeApp  
418 -}  
419 -  
420 -// Delete used to register router for Delete method  
421 -// usage:  
422 -// beego.Delete("/api", func(ctx *context.Context){  
423 -// ctx.Output.Body("hello world")  
424 -// })  
425 -func Delete(rootpath string, f FilterFunc) *App {  
426 - BeeApp.Handlers.Delete(rootpath, f)  
427 - return BeeApp  
428 -}  
429 -  
430 -// Put used to register router for Put method  
431 -// usage:  
432 -// beego.Put("/api", func(ctx *context.Context){  
433 -// ctx.Output.Body("hello world")  
434 -// })  
435 -func Put(rootpath string, f FilterFunc) *App {  
436 - BeeApp.Handlers.Put(rootpath, f)  
437 - return BeeApp  
438 -}  
439 -  
440 -// Head used to register router for Head method  
441 -// usage:  
442 -// beego.Head("/api", func(ctx *context.Context){  
443 -// ctx.Output.Body("hello world")  
444 -// })  
445 -func Head(rootpath string, f FilterFunc) *App {  
446 - BeeApp.Handlers.Head(rootpath, f)  
447 - return BeeApp  
448 -}  
449 -  
450 -// Options used to register router for Options method  
451 -// usage:  
452 -// beego.Options("/api", func(ctx *context.Context){  
453 -// ctx.Output.Body("hello world")  
454 -// })  
455 -func Options(rootpath string, f FilterFunc) *App {  
456 - BeeApp.Handlers.Options(rootpath, f)  
457 - return BeeApp  
458 -}  
459 -  
460 -// Patch used to register router for Patch method  
461 -// usage:  
462 -// beego.Patch("/api", func(ctx *context.Context){  
463 -// ctx.Output.Body("hello world")  
464 -// })  
465 -func Patch(rootpath string, f FilterFunc) *App {  
466 - BeeApp.Handlers.Patch(rootpath, f)  
467 - return BeeApp  
468 -}  
469 -  
470 -// Any used to register router for all methods  
471 -// usage:  
472 -// beego.Any("/api", func(ctx *context.Context){  
473 -// ctx.Output.Body("hello world")  
474 -// })  
475 -func Any(rootpath string, f FilterFunc) *App {  
476 - BeeApp.Handlers.Any(rootpath, f)  
477 - return BeeApp  
478 -}  
479 -  
480 -// Handler used to register a Handler router  
481 -// usage:  
482 -// beego.Handler("/api", http.HandlerFunc(func (w http.ResponseWriter, r *http.Request) {  
483 -// fmt.Fprintf(w, "Hello, %q", html.EscapeString(r.URL.Path))  
484 -// }))  
485 -func Handler(rootpath string, h http.Handler, options ...interface{}) *App {  
486 - BeeApp.Handlers.Handler(rootpath, h, options...)  
487 - return BeeApp  
488 -}  
489 -  
490 -// InsertFilter adds a FilterFunc with pattern condition and action constant.  
491 -// The pos means action constant including  
492 -// beego.BeforeStatic, beego.BeforeRouter, beego.BeforeExec, beego.AfterExec and beego.FinishRouter.  
493 -// The bool params is for setting the returnOnOutput value (false allows multiple filters to execute)  
494 -func InsertFilter(pattern string, pos int, filter FilterFunc, params ...bool) *App {  
495 - BeeApp.Handlers.InsertFilter(pattern, pos, filter, params...)  
496 - return BeeApp  
497 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package beego  
16 -  
17 -import (  
18 - "os"  
19 - "path/filepath"  
20 - "strconv"  
21 - "strings"  
22 -)  
23 -  
24 -const (  
25 - // VERSION represent beego web framework version.  
26 - VERSION = "1.10.0"  
27 -  
28 - // DEV is for develop  
29 - DEV = "dev"  
30 - // PROD is for production  
31 - PROD = "prod"  
32 -)  
33 -  
34 -//hook function to run  
35 -type hookfunc func() error  
36 -  
37 -var (  
38 - hooks = make([]hookfunc, 0) //hook function slice to store the hookfunc  
39 -)  
40 -  
41 -// AddAPPStartHook is used to register the hookfunc  
42 -// The hookfuncs will run in beego.Run()  
43 -// such as initiating session , starting middleware , building template, starting admin control and so on.  
44 -func AddAPPStartHook(hf ...hookfunc) {  
45 - hooks = append(hooks, hf...)  
46 -}  
47 -  
48 -// Run beego application.  
49 -// beego.Run() default run on HttpPort  
50 -// beego.Run("localhost")  
51 -// beego.Run(":8089")  
52 -// beego.Run("127.0.0.1:8089")  
53 -func Run(params ...string) {  
54 -  
55 - initBeforeHTTPRun()  
56 -  
57 - if len(params) > 0 && params[0] != "" {  
58 - strs := strings.Split(params[0], ":")  
59 - if len(strs) > 0 && strs[0] != "" {  
60 - BConfig.Listen.HTTPAddr = strs[0]  
61 - }  
62 - if len(strs) > 1 && strs[1] != "" {  
63 - BConfig.Listen.HTTPPort, _ = strconv.Atoi(strs[1])  
64 - }  
65 -  
66 - BConfig.Listen.Domains = params  
67 - }  
68 -  
69 - BeeApp.Run()  
70 -}  
71 -  
72 -// RunWithMiddleWares Run beego application with middlewares.  
73 -func RunWithMiddleWares(addr string, mws ...MiddleWare) {  
74 - initBeforeHTTPRun()  
75 -  
76 - strs := strings.Split(addr, ":")  
77 - if len(strs) > 0 && strs[0] != "" {  
78 - BConfig.Listen.HTTPAddr = strs[0]  
79 - BConfig.Listen.Domains = []string{strs[0]}  
80 - }  
81 - if len(strs) > 1 && strs[1] != "" {  
82 - BConfig.Listen.HTTPPort, _ = strconv.Atoi(strs[1])  
83 - }  
84 -  
85 - BeeApp.Run(mws...)  
86 -}  
87 -  
88 -func initBeforeHTTPRun() {  
89 - //init hooks  
90 - AddAPPStartHook(  
91 - registerMime,  
92 - registerDefaultErrorHandler,  
93 - registerSession,  
94 - registerTemplate,  
95 - registerAdmin,  
96 - registerGzip,  
97 - )  
98 -  
99 - for _, hk := range hooks {  
100 - if err := hk(); err != nil {  
101 - panic(err)  
102 - }  
103 - }  
104 -}  
105 -  
106 -// TestBeegoInit is for test package init  
107 -func TestBeegoInit(ap string) {  
108 - path := filepath.Join(ap, "conf", "app.conf")  
109 - os.Chdir(ap)  
110 - InitBeegoBeforeTest(path)  
111 -}  
112 -  
113 -// InitBeegoBeforeTest is for test package init  
114 -func InitBeegoBeforeTest(appConfigPath string) {  
115 - if err := LoadAppConfig(appConfigProvider, appConfigPath); err != nil {  
116 - panic(err)  
117 - }  
118 - BConfig.RunMode = "test"  
119 - initBeforeHTTPRun()  
120 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package beego  
16 -  
17 -import (  
18 - "fmt"  
19 - "os"  
20 - "path/filepath"  
21 - "reflect"  
22 - "runtime"  
23 - "strings"  
24 -  
25 - "github.com/astaxie/beego/config"  
26 - "github.com/astaxie/beego/context"  
27 - "github.com/astaxie/beego/logs"  
28 - "github.com/astaxie/beego/session"  
29 - "github.com/astaxie/beego/utils"  
30 -)  
31 -  
32 -// Config is the main struct for BConfig  
33 -type Config struct {  
34 - AppName string //Application name  
35 - RunMode string //Running Mode: dev | prod  
36 - RouterCaseSensitive bool  
37 - ServerName string  
38 - RecoverPanic bool  
39 - RecoverFunc func(*context.Context)  
40 - CopyRequestBody bool  
41 - EnableGzip bool  
42 - MaxMemory int64  
43 - EnableErrorsShow bool  
44 - EnableErrorsRender bool  
45 - Listen Listen  
46 - WebConfig WebConfig  
47 - Log LogConfig  
48 -}  
49 -  
50 -// Listen holds for http and https related config  
51 -type Listen struct {  
52 - Graceful bool // Graceful means use graceful module to start the server  
53 - ServerTimeOut int64  
54 - ListenTCP4 bool  
55 - EnableHTTP bool  
56 - HTTPAddr string  
57 - HTTPPort int  
58 - AutoTLS bool  
59 - Domains []string  
60 - TLSCacheDir string  
61 - EnableHTTPS bool  
62 - EnableMutualHTTPS bool  
63 - HTTPSAddr string  
64 - HTTPSPort int  
65 - HTTPSCertFile string  
66 - HTTPSKeyFile string  
67 - TrustCaFile string  
68 - EnableAdmin bool  
69 - AdminAddr string  
70 - AdminPort int  
71 - EnableFcgi bool  
72 - EnableStdIo bool // EnableStdIo works with EnableFcgi Use FCGI via standard I/O  
73 -}  
74 -  
75 -// WebConfig holds web related config  
76 -type WebConfig struct {  
77 - AutoRender bool  
78 - EnableDocs bool  
79 - FlashName string  
80 - FlashSeparator string  
81 - DirectoryIndex bool  
82 - StaticDir map[string]string  
83 - StaticExtensionsToGzip []string  
84 - TemplateLeft string  
85 - TemplateRight string  
86 - ViewsPath string  
87 - EnableXSRF bool  
88 - XSRFKey string  
89 - XSRFExpire int  
90 - Session SessionConfig  
91 -}  
92 -  
93 -// SessionConfig holds session related config  
94 -type SessionConfig struct {  
95 - SessionOn bool  
96 - SessionProvider string  
97 - SessionName string  
98 - SessionGCMaxLifetime int64  
99 - SessionProviderConfig string  
100 - SessionCookieLifeTime int  
101 - SessionAutoSetCookie bool  
102 - SessionDomain string  
103 - SessionDisableHTTPOnly bool // used to allow for cross domain cookies/javascript cookies.  
104 - SessionEnableSidInHTTPHeader bool // enable store/get the sessionId into/from http headers  
105 - SessionNameInHTTPHeader string  
106 - SessionEnableSidInURLQuery bool // enable get the sessionId from Url Query params  
107 -}  
108 -  
109 -// LogConfig holds Log related config  
110 -type LogConfig struct {  
111 - AccessLogs bool  
112 - EnableStaticLogs bool //log static files requests default: false  
113 - AccessLogsFormat string //access log format: JSON_FORMAT, APACHE_FORMAT or empty string  
114 - FileLineNum bool  
115 - Outputs map[string]string // Store Adaptor : config  
116 -}  
117 -  
118 -var (  
119 - // BConfig is the default config for Application  
120 - BConfig *Config  
121 - // AppConfig is the instance of Config, store the config information from file  
122 - AppConfig *beegoAppConfig  
123 - // AppPath is the absolute path to the app  
124 - AppPath string  
125 - // GlobalSessions is the instance for the session manager  
126 - GlobalSessions *session.Manager  
127 -  
128 - // appConfigPath is the path to the config files  
129 - appConfigPath string  
130 - // appConfigProvider is the provider for the config, default is ini  
131 - appConfigProvider = "ini"  
132 -)  
133 -  
134 -func init() {  
135 - BConfig = newBConfig()  
136 - var err error  
137 - if AppPath, err = filepath.Abs(filepath.Dir(os.Args[0])); err != nil {  
138 - panic(err)  
139 - }  
140 - workPath, err := os.Getwd()  
141 - if err != nil {  
142 - panic(err)  
143 - }  
144 - var filename = "app.conf"  
145 - if os.Getenv("BEEGO_RUNMODE") != "" {  
146 - filename = os.Getenv("BEEGO_RUNMODE") + ".app.conf"  
147 - }  
148 - appConfigPath = filepath.Join(workPath, "conf", filename)  
149 - if !utils.FileExists(appConfigPath) {  
150 - appConfigPath = filepath.Join(AppPath, "conf", filename)  
151 - if !utils.FileExists(appConfigPath) {  
152 - AppConfig = &beegoAppConfig{innerConfig: config.NewFakeConfig()}  
153 - return  
154 - }  
155 - }  
156 - if err = parseConfig(appConfigPath); err != nil {  
157 - panic(err)  
158 - }  
159 -}  
160 -  
161 -func recoverPanic(ctx *context.Context) {  
162 - if err := recover(); err != nil {  
163 - if err == ErrAbort {  
164 - return  
165 - }  
166 - if !BConfig.RecoverPanic {  
167 - panic(err)  
168 - }  
169 - if BConfig.EnableErrorsShow {  
170 - if _, ok := ErrorMaps[fmt.Sprint(err)]; ok {  
171 - exception(fmt.Sprint(err), ctx)  
172 - return  
173 - }  
174 - }  
175 - var stack string  
176 - logs.Critical("the request url is ", ctx.Input.URL())  
177 - logs.Critical("Handler crashed with error", err)  
178 - for i := 1; ; i++ {  
179 - _, file, line, ok := runtime.Caller(i)  
180 - if !ok {  
181 - break  
182 - }  
183 - logs.Critical(fmt.Sprintf("%s:%d", file, line))  
184 - stack = stack + fmt.Sprintln(fmt.Sprintf("%s:%d", file, line))  
185 - }  
186 - if BConfig.RunMode == DEV && BConfig.EnableErrorsRender {  
187 - showErr(err, ctx, stack)  
188 - }  
189 - if ctx.Output.Status != 0 {  
190 - ctx.ResponseWriter.WriteHeader(ctx.Output.Status)  
191 - } else {  
192 - ctx.ResponseWriter.WriteHeader(500)  
193 - }  
194 - }  
195 -}  
196 -  
197 -func newBConfig() *Config {  
198 - return &Config{  
199 - AppName: "beego",  
200 - RunMode: PROD,  
201 - RouterCaseSensitive: true,  
202 - ServerName: "beegoServer:" + VERSION,  
203 - RecoverPanic: true,  
204 - RecoverFunc: recoverPanic,  
205 - CopyRequestBody: false,  
206 - EnableGzip: false,  
207 - MaxMemory: 1 << 26, //64MB  
208 - EnableErrorsShow: true,  
209 - EnableErrorsRender: true,  
210 - Listen: Listen{  
211 - Graceful: false,  
212 - ServerTimeOut: 0,  
213 - ListenTCP4: false,  
214 - EnableHTTP: true,  
215 - AutoTLS: false,  
216 - Domains: []string{},  
217 - TLSCacheDir: ".",  
218 - HTTPAddr: "",  
219 - HTTPPort: 8080,  
220 - EnableHTTPS: false,  
221 - HTTPSAddr: "",  
222 - HTTPSPort: 10443,  
223 - HTTPSCertFile: "",  
224 - HTTPSKeyFile: "",  
225 - EnableAdmin: false,  
226 - AdminAddr: "",  
227 - AdminPort: 8088,  
228 - EnableFcgi: false,  
229 - EnableStdIo: false,  
230 - },  
231 - WebConfig: WebConfig{  
232 - AutoRender: true,  
233 - EnableDocs: false,  
234 - FlashName: "BEEGO_FLASH",  
235 - FlashSeparator: "BEEGOFLASH",  
236 - DirectoryIndex: false,  
237 - StaticDir: map[string]string{"/static": "static"},  
238 - StaticExtensionsToGzip: []string{".css", ".js"},  
239 - TemplateLeft: "{{",  
240 - TemplateRight: "}}",  
241 - ViewsPath: "views",  
242 - EnableXSRF: false,  
243 - XSRFKey: "beegoxsrf",  
244 - XSRFExpire: 0,  
245 - Session: SessionConfig{  
246 - SessionOn: false,  
247 - SessionProvider: "memory",  
248 - SessionName: "beegosessionID",  
249 - SessionGCMaxLifetime: 3600,  
250 - SessionProviderConfig: "",  
251 - SessionDisableHTTPOnly: false,  
252 - SessionCookieLifeTime: 0, //set cookie default is the browser life  
253 - SessionAutoSetCookie: true,  
254 - SessionDomain: "",  
255 - SessionEnableSidInHTTPHeader: false, // enable store/get the sessionId into/from http headers  
256 - SessionNameInHTTPHeader: "Beegosessionid",  
257 - SessionEnableSidInURLQuery: false, // enable get the sessionId from Url Query params  
258 - },  
259 - },  
260 - Log: LogConfig{  
261 - AccessLogs: false,  
262 - EnableStaticLogs: false,  
263 - AccessLogsFormat: "APACHE_FORMAT",  
264 - FileLineNum: true,  
265 - Outputs: map[string]string{"console": ""},  
266 - },  
267 - }  
268 -}  
269 -  
270 -// now only support ini, next will support json.  
271 -func parseConfig(appConfigPath string) (err error) {  
272 - AppConfig, err = newAppConfig(appConfigProvider, appConfigPath)  
273 - if err != nil {  
274 - return err  
275 - }  
276 - return assignConfig(AppConfig)  
277 -}  
278 -  
279 -func assignConfig(ac config.Configer) error {  
280 - for _, i := range []interface{}{BConfig, &BConfig.Listen, &BConfig.WebConfig, &BConfig.Log, &BConfig.WebConfig.Session} {  
281 - assignSingleConfig(i, ac)  
282 - }  
283 - // set the run mode first  
284 - if envRunMode := os.Getenv("BEEGO_RUNMODE"); envRunMode != "" {  
285 - BConfig.RunMode = envRunMode  
286 - } else if runMode := ac.String("RunMode"); runMode != "" {  
287 - BConfig.RunMode = runMode  
288 - }  
289 -  
290 - if sd := ac.String("StaticDir"); sd != "" {  
291 - BConfig.WebConfig.StaticDir = map[string]string{}  
292 - sds := strings.Fields(sd)  
293 - for _, v := range sds {  
294 - if url2fsmap := strings.SplitN(v, ":", 2); len(url2fsmap) == 2 {  
295 - BConfig.WebConfig.StaticDir["/"+strings.Trim(url2fsmap[0], "/")] = url2fsmap[1]  
296 - } else {  
297 - BConfig.WebConfig.StaticDir["/"+strings.Trim(url2fsmap[0], "/")] = url2fsmap[0]  
298 - }  
299 - }  
300 - }  
301 -  
302 - if sgz := ac.String("StaticExtensionsToGzip"); sgz != "" {  
303 - extensions := strings.Split(sgz, ",")  
304 - fileExts := []string{}  
305 - for _, ext := range extensions {  
306 - ext = strings.TrimSpace(ext)  
307 - if ext == "" {  
308 - continue  
309 - }  
310 - if !strings.HasPrefix(ext, ".") {  
311 - ext = "." + ext  
312 - }  
313 - fileExts = append(fileExts, ext)  
314 - }  
315 - if len(fileExts) > 0 {  
316 - BConfig.WebConfig.StaticExtensionsToGzip = fileExts  
317 - }  
318 - }  
319 -  
320 - if lo := ac.String("LogOutputs"); lo != "" {  
321 - // if lo is not nil or empty  
322 - // means user has set his own LogOutputs  
323 - // clear the default setting to BConfig.Log.Outputs  
324 - BConfig.Log.Outputs = make(map[string]string)  
325 - los := strings.Split(lo, ";")  
326 - for _, v := range los {  
327 - if logType2Config := strings.SplitN(v, ",", 2); len(logType2Config) == 2 {  
328 - BConfig.Log.Outputs[logType2Config[0]] = logType2Config[1]  
329 - } else {  
330 - continue  
331 - }  
332 - }  
333 - }  
334 -  
335 - //init log  
336 - logs.Reset()  
337 - for adaptor, config := range BConfig.Log.Outputs {  
338 - err := logs.SetLogger(adaptor, config)  
339 - if err != nil {  
340 - fmt.Fprintln(os.Stderr, fmt.Sprintf("%s with the config %q got err:%s", adaptor, config, err.Error()))  
341 - }  
342 - }  
343 - logs.SetLogFuncCall(BConfig.Log.FileLineNum)  
344 -  
345 - return nil  
346 -}  
347 -  
348 -func assignSingleConfig(p interface{}, ac config.Configer) {  
349 - pt := reflect.TypeOf(p)  
350 - if pt.Kind() != reflect.Ptr {  
351 - return  
352 - }  
353 - pt = pt.Elem()  
354 - if pt.Kind() != reflect.Struct {  
355 - return  
356 - }  
357 - pv := reflect.ValueOf(p).Elem()  
358 -  
359 - for i := 0; i < pt.NumField(); i++ {  
360 - pf := pv.Field(i)  
361 - if !pf.CanSet() {  
362 - continue  
363 - }  
364 - name := pt.Field(i).Name  
365 - switch pf.Kind() {  
366 - case reflect.String:  
367 - pf.SetString(ac.DefaultString(name, pf.String()))  
368 - case reflect.Int, reflect.Int64:  
369 - pf.SetInt(ac.DefaultInt64(name, pf.Int()))  
370 - case reflect.Bool:  
371 - pf.SetBool(ac.DefaultBool(name, pf.Bool()))  
372 - case reflect.Struct:  
373 - default:  
374 - //do nothing here  
375 - }  
376 - }  
377 -  
378 -}  
379 -  
380 -// LoadAppConfig allow developer to apply a config file  
381 -func LoadAppConfig(adapterName, configPath string) error {  
382 - absConfigPath, err := filepath.Abs(configPath)  
383 - if err != nil {  
384 - return err  
385 - }  
386 -  
387 - if !utils.FileExists(absConfigPath) {  
388 - return fmt.Errorf("the target config file: %s don't exist", configPath)  
389 - }  
390 -  
391 - appConfigPath = absConfigPath  
392 - appConfigProvider = adapterName  
393 -  
394 - return parseConfig(appConfigPath)  
395 -}  
396 -  
397 -type beegoAppConfig struct {  
398 - innerConfig config.Configer  
399 -}  
400 -  
401 -func newAppConfig(appConfigProvider, appConfigPath string) (*beegoAppConfig, error) {  
402 - ac, err := config.NewConfig(appConfigProvider, appConfigPath)  
403 - if err != nil {  
404 - return nil, err  
405 - }  
406 - return &beegoAppConfig{ac}, nil  
407 -}  
408 -  
409 -func (b *beegoAppConfig) Set(key, val string) error {  
410 - if err := b.innerConfig.Set(BConfig.RunMode+"::"+key, val); err != nil {  
411 - return err  
412 - }  
413 - return b.innerConfig.Set(key, val)  
414 -}  
415 -  
416 -func (b *beegoAppConfig) String(key string) string {  
417 - if v := b.innerConfig.String(BConfig.RunMode + "::" + key); v != "" {  
418 - return v  
419 - }  
420 - return b.innerConfig.String(key)  
421 -}  
422 -  
423 -func (b *beegoAppConfig) Strings(key string) []string {  
424 - if v := b.innerConfig.Strings(BConfig.RunMode + "::" + key); len(v) > 0 {  
425 - return v  
426 - }  
427 - return b.innerConfig.Strings(key)  
428 -}  
429 -  
430 -func (b *beegoAppConfig) Int(key string) (int, error) {  
431 - if v, err := b.innerConfig.Int(BConfig.RunMode + "::" + key); err == nil {  
432 - return v, nil  
433 - }  
434 - return b.innerConfig.Int(key)  
435 -}  
436 -  
437 -func (b *beegoAppConfig) Int64(key string) (int64, error) {  
438 - if v, err := b.innerConfig.Int64(BConfig.RunMode + "::" + key); err == nil {  
439 - return v, nil  
440 - }  
441 - return b.innerConfig.Int64(key)  
442 -}  
443 -  
444 -func (b *beegoAppConfig) Bool(key string) (bool, error) {  
445 - if v, err := b.innerConfig.Bool(BConfig.RunMode + "::" + key); err == nil {  
446 - return v, nil  
447 - }  
448 - return b.innerConfig.Bool(key)  
449 -}  
450 -  
451 -func (b *beegoAppConfig) Float(key string) (float64, error) {  
452 - if v, err := b.innerConfig.Float(BConfig.RunMode + "::" + key); err == nil {  
453 - return v, nil  
454 - }  
455 - return b.innerConfig.Float(key)  
456 -}  
457 -  
458 -func (b *beegoAppConfig) DefaultString(key string, defaultVal string) string {  
459 - if v := b.String(key); v != "" {  
460 - return v  
461 - }  
462 - return defaultVal  
463 -}  
464 -  
465 -func (b *beegoAppConfig) DefaultStrings(key string, defaultVal []string) []string {  
466 - if v := b.Strings(key); len(v) != 0 {  
467 - return v  
468 - }  
469 - return defaultVal  
470 -}  
471 -  
472 -func (b *beegoAppConfig) DefaultInt(key string, defaultVal int) int {  
473 - if v, err := b.Int(key); err == nil {  
474 - return v  
475 - }  
476 - return defaultVal  
477 -}  
478 -  
479 -func (b *beegoAppConfig) DefaultInt64(key string, defaultVal int64) int64 {  
480 - if v, err := b.Int64(key); err == nil {  
481 - return v  
482 - }  
483 - return defaultVal  
484 -}  
485 -  
486 -func (b *beegoAppConfig) DefaultBool(key string, defaultVal bool) bool {  
487 - if v, err := b.Bool(key); err == nil {  
488 - return v  
489 - }  
490 - return defaultVal  
491 -}  
492 -  
493 -func (b *beegoAppConfig) DefaultFloat(key string, defaultVal float64) float64 {  
494 - if v, err := b.Float(key); err == nil {  
495 - return v  
496 - }  
497 - return defaultVal  
498 -}  
499 -  
500 -func (b *beegoAppConfig) DIY(key string) (interface{}, error) {  
501 - return b.innerConfig.DIY(key)  
502 -}  
503 -  
504 -func (b *beegoAppConfig) GetSection(section string) (map[string]string, error) {  
505 - return b.innerConfig.GetSection(section)  
506 -}  
507 -  
508 -func (b *beegoAppConfig) SaveConfigFile(filename string) error {  
509 - return b.innerConfig.SaveConfigFile(filename)  
510 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -// Package config is used to parse config.  
16 -// Usage:  
17 -// import "github.com/astaxie/beego/config"  
18 -//Examples.  
19 -//  
20 -// cnf, err := config.NewConfig("ini", "config.conf")  
21 -//  
22 -// cnf APIS:  
23 -//  
24 -// cnf.Set(key, val string) error  
25 -// cnf.String(key string) string  
26 -// cnf.Strings(key string) []string  
27 -// cnf.Int(key string) (int, error)  
28 -// cnf.Int64(key string) (int64, error)  
29 -// cnf.Bool(key string) (bool, error)  
30 -// cnf.Float(key string) (float64, error)  
31 -// cnf.DefaultString(key string, defaultVal string) string  
32 -// cnf.DefaultStrings(key string, defaultVal []string) []string  
33 -// cnf.DefaultInt(key string, defaultVal int) int  
34 -// cnf.DefaultInt64(key string, defaultVal int64) int64  
35 -// cnf.DefaultBool(key string, defaultVal bool) bool  
36 -// cnf.DefaultFloat(key string, defaultVal float64) float64  
37 -// cnf.DIY(key string) (interface{}, error)  
38 -// cnf.GetSection(section string) (map[string]string, error)  
39 -// cnf.SaveConfigFile(filename string) error  
40 -//More docs http://beego.me/docs/module/config.md  
41 -package config  
42 -  
43 -import (  
44 - "fmt"  
45 - "os"  
46 - "reflect"  
47 - "time"  
48 -)  
49 -  
50 -// Configer defines how to get and set value from configuration raw data.  
51 -type Configer interface {  
52 - Set(key, val string) error //support section::key type in given key when using ini type.  
53 - String(key string) string //support section::key type in key string when using ini and json type; Int,Int64,Bool,Float,DIY are same.  
54 - Strings(key string) []string //get string slice  
55 - Int(key string) (int, error)  
56 - Int64(key string) (int64, error)  
57 - Bool(key string) (bool, error)  
58 - Float(key string) (float64, error)  
59 - DefaultString(key string, defaultVal string) string // support section::key type in key string when using ini and json type; Int,Int64,Bool,Float,DIY are same.  
60 - DefaultStrings(key string, defaultVal []string) []string //get string slice  
61 - DefaultInt(key string, defaultVal int) int  
62 - DefaultInt64(key string, defaultVal int64) int64  
63 - DefaultBool(key string, defaultVal bool) bool  
64 - DefaultFloat(key string, defaultVal float64) float64  
65 - DIY(key string) (interface{}, error)  
66 - GetSection(section string) (map[string]string, error)  
67 - SaveConfigFile(filename string) error  
68 -}  
69 -  
70 -// Config is the adapter interface for parsing config file to get raw data to Configer.  
71 -type Config interface {  
72 - Parse(key string) (Configer, error)  
73 - ParseData(data []byte) (Configer, error)  
74 -}  
75 -  
76 -var adapters = make(map[string]Config)  
77 -  
78 -// Register makes a config adapter available by the adapter name.  
79 -// If Register is called twice with the same name or if driver is nil,  
80 -// it panics.  
81 -func Register(name string, adapter Config) {  
82 - if adapter == nil {  
83 - panic("config: Register adapter is nil")  
84 - }  
85 - if _, ok := adapters[name]; ok {  
86 - panic("config: Register called twice for adapter " + name)  
87 - }  
88 - adapters[name] = adapter  
89 -}  
90 -  
91 -// NewConfig adapterName is ini/json/xml/yaml.  
92 -// filename is the config file path.  
93 -func NewConfig(adapterName, filename string) (Configer, error) {  
94 - adapter, ok := adapters[adapterName]  
95 - if !ok {  
96 - return nil, fmt.Errorf("config: unknown adaptername %q (forgotten import?)", adapterName)  
97 - }  
98 - return adapter.Parse(filename)  
99 -}  
100 -  
101 -// NewConfigData adapterName is ini/json/xml/yaml.  
102 -// data is the config data.  
103 -func NewConfigData(adapterName string, data []byte) (Configer, error) {  
104 - adapter, ok := adapters[adapterName]  
105 - if !ok {  
106 - return nil, fmt.Errorf("config: unknown adaptername %q (forgotten import?)", adapterName)  
107 - }  
108 - return adapter.ParseData(data)  
109 -}  
110 -  
111 -// ExpandValueEnvForMap convert all string value with environment variable.  
112 -func ExpandValueEnvForMap(m map[string]interface{}) map[string]interface{} {  
113 - for k, v := range m {  
114 - switch value := v.(type) {  
115 - case string:  
116 - m[k] = ExpandValueEnv(value)  
117 - case map[string]interface{}:  
118 - m[k] = ExpandValueEnvForMap(value)  
119 - case map[string]string:  
120 - for k2, v2 := range value {  
121 - value[k2] = ExpandValueEnv(v2)  
122 - }  
123 - m[k] = value  
124 - }  
125 - }  
126 - return m  
127 -}  
128 -  
129 -// ExpandValueEnv returns value of convert with environment variable.  
130 -//  
131 -// Return environment variable if value start with "${" and end with "}".  
132 -// Return default value if environment variable is empty or not exist.  
133 -//  
134 -// It accept value formats "${env}" , "${env||}}" , "${env||defaultValue}" , "defaultvalue".  
135 -// Examples:  
136 -// v1 := config.ExpandValueEnv("${GOPATH}") // return the GOPATH environment variable.  
137 -// v2 := config.ExpandValueEnv("${GOAsta||/usr/local/go}") // return the default value "/usr/local/go/".  
138 -// v3 := config.ExpandValueEnv("Astaxie") // return the value "Astaxie".  
139 -func ExpandValueEnv(value string) (realValue string) {  
140 - realValue = value  
141 -  
142 - vLen := len(value)  
143 - // 3 = ${}  
144 - if vLen < 3 {  
145 - return  
146 - }  
147 - // Need start with "${" and end with "}", then return.  
148 - if value[0] != '$' || value[1] != '{' || value[vLen-1] != '}' {  
149 - return  
150 - }  
151 -  
152 - key := ""  
153 - defaultV := ""  
154 - // value start with "${"  
155 - for i := 2; i < vLen; i++ {  
156 - if value[i] == '|' && (i+1 < vLen && value[i+1] == '|') {  
157 - key = value[2:i]  
158 - defaultV = value[i+2 : vLen-1] // other string is default value.  
159 - break  
160 - } else if value[i] == '}' {  
161 - key = value[2:i]  
162 - break  
163 - }  
164 - }  
165 -  
166 - realValue = os.Getenv(key)  
167 - if realValue == "" {  
168 - realValue = defaultV  
169 - }  
170 -  
171 - return  
172 -}  
173 -  
174 -// ParseBool returns the boolean value represented by the string.  
175 -//  
176 -// It accepts 1, 1.0, t, T, TRUE, true, True, YES, yes, Yes,Y, y, ON, on, On,  
177 -// 0, 0.0, f, F, FALSE, false, False, NO, no, No, N,n, OFF, off, Off.  
178 -// Any other value returns an error.  
179 -func ParseBool(val interface{}) (value bool, err error) {  
180 - if val != nil {  
181 - switch v := val.(type) {  
182 - case bool:  
183 - return v, nil  
184 - case string:  
185 - switch v {  
186 - case "1", "t", "T", "true", "TRUE", "True", "YES", "yes", "Yes", "Y", "y", "ON", "on", "On":  
187 - return true, nil  
188 - case "0", "f", "F", "false", "FALSE", "False", "NO", "no", "No", "N", "n", "OFF", "off", "Off":  
189 - return false, nil  
190 - }  
191 - case int8, int32, int64:  
192 - strV := fmt.Sprintf("%d", v)  
193 - if strV == "1" {  
194 - return true, nil  
195 - } else if strV == "0" {  
196 - return false, nil  
197 - }  
198 - case float64:  
199 - if v == 1.0 {  
200 - return true, nil  
201 - } else if v == 0.0 {  
202 - return false, nil  
203 - }  
204 - }  
205 - return false, fmt.Errorf("parsing %q: invalid syntax", val)  
206 - }  
207 - return false, fmt.Errorf("parsing <nil>: invalid syntax")  
208 -}  
209 -  
210 -// ToString converts values of any type to string.  
211 -func ToString(x interface{}) string {  
212 - switch y := x.(type) {  
213 -  
214 - // Handle dates with special logic  
215 - // This needs to come above the fmt.Stringer  
216 - // test since time.Time's have a .String()  
217 - // method  
218 - case time.Time:  
219 - return y.Format("A Monday")  
220 -  
221 - // Handle type string  
222 - case string:  
223 - return y  
224 -  
225 - // Handle type with .String() method  
226 - case fmt.Stringer:  
227 - return y.String()  
228 -  
229 - // Handle type with .Error() method  
230 - case error:  
231 - return y.Error()  
232 -  
233 - }  
234 -  
235 - // Handle named string type  
236 - if v := reflect.ValueOf(x); v.Kind() == reflect.String {  
237 - return v.String()  
238 - }  
239 -  
240 - // Fallback to fmt package for anything else like numeric types  
241 - return fmt.Sprint(x)  
242 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package config  
16 -  
17 -import (  
18 - "errors"  
19 - "strconv"  
20 - "strings"  
21 -)  
22 -  
23 -type fakeConfigContainer struct {  
24 - data map[string]string  
25 -}  
26 -  
27 -func (c *fakeConfigContainer) getData(key string) string {  
28 - return c.data[strings.ToLower(key)]  
29 -}  
30 -  
31 -func (c *fakeConfigContainer) Set(key, val string) error {  
32 - c.data[strings.ToLower(key)] = val  
33 - return nil  
34 -}  
35 -  
36 -func (c *fakeConfigContainer) String(key string) string {  
37 - return c.getData(key)  
38 -}  
39 -  
40 -func (c *fakeConfigContainer) DefaultString(key string, defaultval string) string {  
41 - v := c.String(key)  
42 - if v == "" {  
43 - return defaultval  
44 - }  
45 - return v  
46 -}  
47 -  
48 -func (c *fakeConfigContainer) Strings(key string) []string {  
49 - v := c.String(key)  
50 - if v == "" {  
51 - return nil  
52 - }  
53 - return strings.Split(v, ";")  
54 -}  
55 -  
56 -func (c *fakeConfigContainer) DefaultStrings(key string, defaultval []string) []string {  
57 - v := c.Strings(key)  
58 - if v == nil {  
59 - return defaultval  
60 - }  
61 - return v  
62 -}  
63 -  
64 -func (c *fakeConfigContainer) Int(key string) (int, error) {  
65 - return strconv.Atoi(c.getData(key))  
66 -}  
67 -  
68 -func (c *fakeConfigContainer) DefaultInt(key string, defaultval int) int {  
69 - v, err := c.Int(key)  
70 - if err != nil {  
71 - return defaultval  
72 - }  
73 - return v  
74 -}  
75 -  
76 -func (c *fakeConfigContainer) Int64(key string) (int64, error) {  
77 - return strconv.ParseInt(c.getData(key), 10, 64)  
78 -}  
79 -  
80 -func (c *fakeConfigContainer) DefaultInt64(key string, defaultval int64) int64 {  
81 - v, err := c.Int64(key)  
82 - if err != nil {  
83 - return defaultval  
84 - }  
85 - return v  
86 -}  
87 -  
88 -func (c *fakeConfigContainer) Bool(key string) (bool, error) {  
89 - return ParseBool(c.getData(key))  
90 -}  
91 -  
92 -func (c *fakeConfigContainer) DefaultBool(key string, defaultval bool) bool {  
93 - v, err := c.Bool(key)  
94 - if err != nil {  
95 - return defaultval  
96 - }  
97 - return v  
98 -}  
99 -  
100 -func (c *fakeConfigContainer) Float(key string) (float64, error) {  
101 - return strconv.ParseFloat(c.getData(key), 64)  
102 -}  
103 -  
104 -func (c *fakeConfigContainer) DefaultFloat(key string, defaultval float64) float64 {  
105 - v, err := c.Float(key)  
106 - if err != nil {  
107 - return defaultval  
108 - }  
109 - return v  
110 -}  
111 -  
112 -func (c *fakeConfigContainer) DIY(key string) (interface{}, error) {  
113 - if v, ok := c.data[strings.ToLower(key)]; ok {  
114 - return v, nil  
115 - }  
116 - return nil, errors.New("key not find")  
117 -}  
118 -  
119 -func (c *fakeConfigContainer) GetSection(section string) (map[string]string, error) {  
120 - return nil, errors.New("not implement in the fakeConfigContainer")  
121 -}  
122 -  
123 -func (c *fakeConfigContainer) SaveConfigFile(filename string) error {  
124 - return errors.New("not implement in the fakeConfigContainer")  
125 -}  
126 -  
127 -var _ Configer = new(fakeConfigContainer)  
128 -  
129 -// NewFakeConfig return a fake Configer  
130 -func NewFakeConfig() Configer {  
131 - return &fakeConfigContainer{  
132 - data: make(map[string]string),  
133 - }  
134 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package config  
16 -  
17 -import (  
18 - "bufio"  
19 - "bytes"  
20 - "errors"  
21 - "io"  
22 - "io/ioutil"  
23 - "os"  
24 - "os/user"  
25 - "path/filepath"  
26 - "strconv"  
27 - "strings"  
28 - "sync"  
29 -)  
30 -  
31 -var (  
32 - defaultSection = "default" // default section means if some ini items not in a section, make them in default section,  
33 - bNumComment = []byte{'#'} // number signal  
34 - bSemComment = []byte{';'} // semicolon signal  
35 - bEmpty = []byte{}  
36 - bEqual = []byte{'='} // equal signal  
37 - bDQuote = []byte{'"'} // quote signal  
38 - sectionStart = []byte{'['} // section start signal  
39 - sectionEnd = []byte{']'} // section end signal  
40 - lineBreak = "\n"  
41 -)  
42 -  
43 -// IniConfig implements Config to parse ini file.  
44 -type IniConfig struct {  
45 -}  
46 -  
47 -// Parse creates a new Config and parses the file configuration from the named file.  
48 -func (ini *IniConfig) Parse(name string) (Configer, error) {  
49 - return ini.parseFile(name)  
50 -}  
51 -  
52 -func (ini *IniConfig) parseFile(name string) (*IniConfigContainer, error) {  
53 - data, err := ioutil.ReadFile(name)  
54 - if err != nil {  
55 - return nil, err  
56 - }  
57 -  
58 - return ini.parseData(filepath.Dir(name), data)  
59 -}  
60 -  
61 -func (ini *IniConfig) parseData(dir string, data []byte) (*IniConfigContainer, error) {  
62 - cfg := &IniConfigContainer{  
63 - data: make(map[string]map[string]string),  
64 - sectionComment: make(map[string]string),  
65 - keyComment: make(map[string]string),  
66 - RWMutex: sync.RWMutex{},  
67 - }  
68 - cfg.Lock()  
69 - defer cfg.Unlock()  
70 -  
71 - var comment bytes.Buffer  
72 - buf := bufio.NewReader(bytes.NewBuffer(data))  
73 - // check the BOM  
74 - head, err := buf.Peek(3)  
75 - if err == nil && head[0] == 239 && head[1] == 187 && head[2] == 191 {  
76 - for i := 1; i <= 3; i++ {  
77 - buf.ReadByte()  
78 - }  
79 - }  
80 - section := defaultSection  
81 - for {  
82 - line, _, err := buf.ReadLine()  
83 - if err == io.EOF {  
84 - break  
85 - }  
86 - //It might be a good idea to throw a error on all unknonw errors?  
87 - if _, ok := err.(*os.PathError); ok {  
88 - return nil, err  
89 - }  
90 - line = bytes.TrimSpace(line)  
91 - if bytes.Equal(line, bEmpty) {  
92 - continue  
93 - }  
94 - var bComment []byte  
95 - switch {  
96 - case bytes.HasPrefix(line, bNumComment):  
97 - bComment = bNumComment  
98 - case bytes.HasPrefix(line, bSemComment):  
99 - bComment = bSemComment  
100 - }  
101 - if bComment != nil {  
102 - line = bytes.TrimLeft(line, string(bComment))  
103 - // Need append to a new line if multi-line comments.  
104 - if comment.Len() > 0 {  
105 - comment.WriteByte('\n')  
106 - }  
107 - comment.Write(line)  
108 - continue  
109 - }  
110 -  
111 - if bytes.HasPrefix(line, sectionStart) && bytes.HasSuffix(line, sectionEnd) {  
112 - section = strings.ToLower(string(line[1 : len(line)-1])) // section name case insensitive  
113 - if comment.Len() > 0 {  
114 - cfg.sectionComment[section] = comment.String()  
115 - comment.Reset()  
116 - }  
117 - if _, ok := cfg.data[section]; !ok {  
118 - cfg.data[section] = make(map[string]string)  
119 - }  
120 - continue  
121 - }  
122 -  
123 - if _, ok := cfg.data[section]; !ok {  
124 - cfg.data[section] = make(map[string]string)  
125 - }  
126 - keyValue := bytes.SplitN(line, bEqual, 2)  
127 -  
128 - key := string(bytes.TrimSpace(keyValue[0])) // key name case insensitive  
129 - key = strings.ToLower(key)  
130 -  
131 - // handle include "other.conf"  
132 - if len(keyValue) == 1 && strings.HasPrefix(key, "include") {  
133 -  
134 - includefiles := strings.Fields(key)  
135 - if includefiles[0] == "include" && len(includefiles) == 2 {  
136 -  
137 - otherfile := strings.Trim(includefiles[1], "\"")  
138 - if !filepath.IsAbs(otherfile) {  
139 - otherfile = filepath.Join(dir, otherfile)  
140 - }  
141 -  
142 - i, err := ini.parseFile(otherfile)  
143 - if err != nil {  
144 - return nil, err  
145 - }  
146 -  
147 - for sec, dt := range i.data {  
148 - if _, ok := cfg.data[sec]; !ok {  
149 - cfg.data[sec] = make(map[string]string)  
150 - }  
151 - for k, v := range dt {  
152 - cfg.data[sec][k] = v  
153 - }  
154 - }  
155 -  
156 - for sec, comm := range i.sectionComment {  
157 - cfg.sectionComment[sec] = comm  
158 - }  
159 -  
160 - for k, comm := range i.keyComment {  
161 - cfg.keyComment[k] = comm  
162 - }  
163 -  
164 - continue  
165 - }  
166 - }  
167 -  
168 - if len(keyValue) != 2 {  
169 - return nil, errors.New("read the content error: \"" + string(line) + "\", should key = val")  
170 - }  
171 - val := bytes.TrimSpace(keyValue[1])  
172 - if bytes.HasPrefix(val, bDQuote) {  
173 - val = bytes.Trim(val, `"`)  
174 - }  
175 -  
176 - cfg.data[section][key] = ExpandValueEnv(string(val))  
177 - if comment.Len() > 0 {  
178 - cfg.keyComment[section+"."+key] = comment.String()  
179 - comment.Reset()  
180 - }  
181 -  
182 - }  
183 - return cfg, nil  
184 -}  
185 -  
186 -// ParseData parse ini the data  
187 -// When include other.conf,other.conf is either absolute directory  
188 -// or under beego in default temporary directory(/tmp/beego[-username]).  
189 -func (ini *IniConfig) ParseData(data []byte) (Configer, error) {  
190 - dir := "beego"  
191 - currentUser, err := user.Current()  
192 - if err == nil {  
193 - dir = "beego-" + currentUser.Username  
194 - }  
195 - dir = filepath.Join(os.TempDir(), dir)  
196 - if err = os.MkdirAll(dir, os.ModePerm); err != nil {  
197 - return nil, err  
198 - }  
199 -  
200 - return ini.parseData(dir, data)  
201 -}  
202 -  
203 -// IniConfigContainer A Config represents the ini configuration.  
204 -// When set and get value, support key as section:name type.  
205 -type IniConfigContainer struct {  
206 - data map[string]map[string]string // section=> key:val  
207 - sectionComment map[string]string // section : comment  
208 - keyComment map[string]string // id: []{comment, key...}; id 1 is for main comment.  
209 - sync.RWMutex  
210 -}  
211 -  
212 -// Bool returns the boolean value for a given key.  
213 -func (c *IniConfigContainer) Bool(key string) (bool, error) {  
214 - return ParseBool(c.getdata(key))  
215 -}  
216 -  
217 -// DefaultBool returns the boolean value for a given key.  
218 -// if err != nil return defaultval  
219 -func (c *IniConfigContainer) DefaultBool(key string, defaultval bool) bool {  
220 - v, err := c.Bool(key)  
221 - if err != nil {  
222 - return defaultval  
223 - }  
224 - return v  
225 -}  
226 -  
227 -// Int returns the integer value for a given key.  
228 -func (c *IniConfigContainer) Int(key string) (int, error) {  
229 - return strconv.Atoi(c.getdata(key))  
230 -}  
231 -  
232 -// DefaultInt returns the integer value for a given key.  
233 -// if err != nil return defaultval  
234 -func (c *IniConfigContainer) DefaultInt(key string, defaultval int) int {  
235 - v, err := c.Int(key)  
236 - if err != nil {  
237 - return defaultval  
238 - }  
239 - return v  
240 -}  
241 -  
242 -// Int64 returns the int64 value for a given key.  
243 -func (c *IniConfigContainer) Int64(key string) (int64, error) {  
244 - return strconv.ParseInt(c.getdata(key), 10, 64)  
245 -}  
246 -  
247 -// DefaultInt64 returns the int64 value for a given key.  
248 -// if err != nil return defaultval  
249 -func (c *IniConfigContainer) DefaultInt64(key string, defaultval int64) int64 {  
250 - v, err := c.Int64(key)  
251 - if err != nil {  
252 - return defaultval  
253 - }  
254 - return v  
255 -}  
256 -  
257 -// Float returns the float value for a given key.  
258 -func (c *IniConfigContainer) Float(key string) (float64, error) {  
259 - return strconv.ParseFloat(c.getdata(key), 64)  
260 -}  
261 -  
262 -// DefaultFloat returns the float64 value for a given key.  
263 -// if err != nil return defaultval  
264 -func (c *IniConfigContainer) DefaultFloat(key string, defaultval float64) float64 {  
265 - v, err := c.Float(key)  
266 - if err != nil {  
267 - return defaultval  
268 - }  
269 - return v  
270 -}  
271 -  
272 -// String returns the string value for a given key.  
273 -func (c *IniConfigContainer) String(key string) string {  
274 - return c.getdata(key)  
275 -}  
276 -  
277 -// DefaultString returns the string value for a given key.  
278 -// if err != nil return defaultval  
279 -func (c *IniConfigContainer) DefaultString(key string, defaultval string) string {  
280 - v := c.String(key)  
281 - if v == "" {  
282 - return defaultval  
283 - }  
284 - return v  
285 -}  
286 -  
287 -// Strings returns the []string value for a given key.  
288 -// Return nil if config value does not exist or is empty.  
289 -func (c *IniConfigContainer) Strings(key string) []string {  
290 - v := c.String(key)  
291 - if v == "" {  
292 - return nil  
293 - }  
294 - return strings.Split(v, ";")  
295 -}  
296 -  
297 -// DefaultStrings returns the []string value for a given key.  
298 -// if err != nil return defaultval  
299 -func (c *IniConfigContainer) DefaultStrings(key string, defaultval []string) []string {  
300 - v := c.Strings(key)  
301 - if v == nil {  
302 - return defaultval  
303 - }  
304 - return v  
305 -}  
306 -  
307 -// GetSection returns map for the given section  
308 -func (c *IniConfigContainer) GetSection(section string) (map[string]string, error) {  
309 - if v, ok := c.data[section]; ok {  
310 - return v, nil  
311 - }  
312 - return nil, errors.New("not exist section")  
313 -}  
314 -  
315 -// SaveConfigFile save the config into file.  
316 -//  
317 -// BUG(env): The environment variable config item will be saved with real value in SaveConfigFile Function.  
318 -func (c *IniConfigContainer) SaveConfigFile(filename string) (err error) {  
319 - // Write configuration file by filename.  
320 - f, err := os.Create(filename)  
321 - if err != nil {  
322 - return err  
323 - }  
324 - defer f.Close()  
325 -  
326 - // Get section or key comments. Fixed #1607  
327 - getCommentStr := func(section, key string) string {  
328 - var (  
329 - comment string  
330 - ok bool  
331 - )  
332 - if len(key) == 0 {  
333 - comment, ok = c.sectionComment[section]  
334 - } else {  
335 - comment, ok = c.keyComment[section+"."+key]  
336 - }  
337 -  
338 - if ok {  
339 - // Empty comment  
340 - if len(comment) == 0 || len(strings.TrimSpace(comment)) == 0 {  
341 - return string(bNumComment)  
342 - }  
343 - prefix := string(bNumComment)  
344 - // Add the line head character "#"  
345 - return prefix + strings.Replace(comment, lineBreak, lineBreak+prefix, -1)  
346 - }  
347 - return ""  
348 - }  
349 -  
350 - buf := bytes.NewBuffer(nil)  
351 - // Save default section at first place  
352 - if dt, ok := c.data[defaultSection]; ok {  
353 - for key, val := range dt {  
354 - if key != " " {  
355 - // Write key comments.  
356 - if v := getCommentStr(defaultSection, key); len(v) > 0 {  
357 - if _, err = buf.WriteString(v + lineBreak); err != nil {  
358 - return err  
359 - }  
360 - }  
361 -  
362 - // Write key and value.  
363 - if _, err = buf.WriteString(key + string(bEqual) + val + lineBreak); err != nil {  
364 - return err  
365 - }  
366 - }  
367 - }  
368 -  
369 - // Put a line between sections.  
370 - if _, err = buf.WriteString(lineBreak); err != nil {  
371 - return err  
372 - }  
373 - }  
374 - // Save named sections  
375 - for section, dt := range c.data {  
376 - if section != defaultSection {  
377 - // Write section comments.  
378 - if v := getCommentStr(section, ""); len(v) > 0 {  
379 - if _, err = buf.WriteString(v + lineBreak); err != nil {  
380 - return err  
381 - }  
382 - }  
383 -  
384 - // Write section name.  
385 - if _, err = buf.WriteString(string(sectionStart) + section + string(sectionEnd) + lineBreak); err != nil {  
386 - return err  
387 - }  
388 -  
389 - for key, val := range dt {  
390 - if key != " " {  
391 - // Write key comments.  
392 - if v := getCommentStr(section, key); len(v) > 0 {  
393 - if _, err = buf.WriteString(v + lineBreak); err != nil {  
394 - return err  
395 - }  
396 - }  
397 -  
398 - // Write key and value.  
399 - if _, err = buf.WriteString(key + string(bEqual) + val + lineBreak); err != nil {  
400 - return err  
401 - }  
402 - }  
403 - }  
404 -  
405 - // Put a line between sections.  
406 - if _, err = buf.WriteString(lineBreak); err != nil {  
407 - return err  
408 - }  
409 - }  
410 - }  
411 - _, err = buf.WriteTo(f)  
412 - return err  
413 -}  
414 -  
415 -// Set writes a new value for key.  
416 -// if write to one section, the key need be "section::key".  
417 -// if the section is not existed, it panics.  
418 -func (c *IniConfigContainer) Set(key, value string) error {  
419 - c.Lock()  
420 - defer c.Unlock()  
421 - if len(key) == 0 {  
422 - return errors.New("key is empty")  
423 - }  
424 -  
425 - var (  
426 - section, k string  
427 - sectionKey = strings.Split(strings.ToLower(key), "::")  
428 - )  
429 -  
430 - if len(sectionKey) >= 2 {  
431 - section = sectionKey[0]  
432 - k = sectionKey[1]  
433 - } else {  
434 - section = defaultSection  
435 - k = sectionKey[0]  
436 - }  
437 -  
438 - if _, ok := c.data[section]; !ok {  
439 - c.data[section] = make(map[string]string)  
440 - }  
441 - c.data[section][k] = value  
442 - return nil  
443 -}  
444 -  
445 -// DIY returns the raw value by a given key.  
446 -func (c *IniConfigContainer) DIY(key string) (v interface{}, err error) {  
447 - if v, ok := c.data[strings.ToLower(key)]; ok {  
448 - return v, nil  
449 - }  
450 - return v, errors.New("key not find")  
451 -}  
452 -  
453 -// section.key or key  
454 -func (c *IniConfigContainer) getdata(key string) string {  
455 - if len(key) == 0 {  
456 - return ""  
457 - }  
458 - c.RLock()  
459 - defer c.RUnlock()  
460 -  
461 - var (  
462 - section, k string  
463 - sectionKey = strings.Split(strings.ToLower(key), "::")  
464 - )  
465 - if len(sectionKey) >= 2 {  
466 - section = sectionKey[0]  
467 - k = sectionKey[1]  
468 - } else {  
469 - section = defaultSection  
470 - k = sectionKey[0]  
471 - }  
472 - if v, ok := c.data[section]; ok {  
473 - if vv, ok := v[k]; ok {  
474 - return vv  
475 - }  
476 - }  
477 - return ""  
478 -}  
479 -  
480 -func init() {  
481 - Register("ini", &IniConfig{})  
482 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package config  
16 -  
17 -import (  
18 - "encoding/json"  
19 - "errors"  
20 - "fmt"  
21 - "io/ioutil"  
22 - "os"  
23 - "strings"  
24 - "sync"  
25 -)  
26 -  
27 -// JSONConfig is a json config parser and implements Config interface.  
28 -type JSONConfig struct {  
29 -}  
30 -  
31 -// Parse returns a ConfigContainer with parsed json config map.  
32 -func (js *JSONConfig) Parse(filename string) (Configer, error) {  
33 - file, err := os.Open(filename)  
34 - if err != nil {  
35 - return nil, err  
36 - }  
37 - defer file.Close()  
38 - content, err := ioutil.ReadAll(file)  
39 - if err != nil {  
40 - return nil, err  
41 - }  
42 -  
43 - return js.ParseData(content)  
44 -}  
45 -  
46 -// ParseData returns a ConfigContainer with json string  
47 -func (js *JSONConfig) ParseData(data []byte) (Configer, error) {  
48 - x := &JSONConfigContainer{  
49 - data: make(map[string]interface{}),  
50 - }  
51 - err := json.Unmarshal(data, &x.data)  
52 - if err != nil {  
53 - var wrappingArray []interface{}  
54 - err2 := json.Unmarshal(data, &wrappingArray)  
55 - if err2 != nil {  
56 - return nil, err  
57 - }  
58 - x.data["rootArray"] = wrappingArray  
59 - }  
60 -  
61 - x.data = ExpandValueEnvForMap(x.data)  
62 -  
63 - return x, nil  
64 -}  
65 -  
66 -// JSONConfigContainer A Config represents the json configuration.  
67 -// Only when get value, support key as section:name type.  
68 -type JSONConfigContainer struct {  
69 - data map[string]interface{}  
70 - sync.RWMutex  
71 -}  
72 -  
73 -// Bool returns the boolean value for a given key.  
74 -func (c *JSONConfigContainer) Bool(key string) (bool, error) {  
75 - val := c.getData(key)  
76 - if val != nil {  
77 - return ParseBool(val)  
78 - }  
79 - return false, fmt.Errorf("not exist key: %q", key)  
80 -}  
81 -  
82 -// DefaultBool return the bool value if has no error  
83 -// otherwise return the defaultval  
84 -func (c *JSONConfigContainer) DefaultBool(key string, defaultval bool) bool {  
85 - if v, err := c.Bool(key); err == nil {  
86 - return v  
87 - }  
88 - return defaultval  
89 -}  
90 -  
91 -// Int returns the integer value for a given key.  
92 -func (c *JSONConfigContainer) Int(key string) (int, error) {  
93 - val := c.getData(key)  
94 - if val != nil {  
95 - if v, ok := val.(float64); ok {  
96 - return int(v), nil  
97 - }  
98 - return 0, errors.New("not int value")  
99 - }  
100 - return 0, errors.New("not exist key:" + key)  
101 -}  
102 -  
103 -// DefaultInt returns the integer value for a given key.  
104 -// if err != nil return defaultval  
105 -func (c *JSONConfigContainer) DefaultInt(key string, defaultval int) int {  
106 - if v, err := c.Int(key); err == nil {  
107 - return v  
108 - }  
109 - return defaultval  
110 -}  
111 -  
112 -// Int64 returns the int64 value for a given key.  
113 -func (c *JSONConfigContainer) Int64(key string) (int64, error) {  
114 - val := c.getData(key)  
115 - if val != nil {  
116 - if v, ok := val.(float64); ok {  
117 - return int64(v), nil  
118 - }  
119 - return 0, errors.New("not int64 value")  
120 - }  
121 - return 0, errors.New("not exist key:" + key)  
122 -}  
123 -  
124 -// DefaultInt64 returns the int64 value for a given key.  
125 -// if err != nil return defaultval  
126 -func (c *JSONConfigContainer) DefaultInt64(key string, defaultval int64) int64 {  
127 - if v, err := c.Int64(key); err == nil {  
128 - return v  
129 - }  
130 - return defaultval  
131 -}  
132 -  
133 -// Float returns the float value for a given key.  
134 -func (c *JSONConfigContainer) Float(key string) (float64, error) {  
135 - val := c.getData(key)  
136 - if val != nil {  
137 - if v, ok := val.(float64); ok {  
138 - return v, nil  
139 - }  
140 - return 0.0, errors.New("not float64 value")  
141 - }  
142 - return 0.0, errors.New("not exist key:" + key)  
143 -}  
144 -  
145 -// DefaultFloat returns the float64 value for a given key.  
146 -// if err != nil return defaultval  
147 -func (c *JSONConfigContainer) DefaultFloat(key string, defaultval float64) float64 {  
148 - if v, err := c.Float(key); err == nil {  
149 - return v  
150 - }  
151 - return defaultval  
152 -}  
153 -  
154 -// String returns the string value for a given key.  
155 -func (c *JSONConfigContainer) String(key string) string {  
156 - val := c.getData(key)  
157 - if val != nil {  
158 - if v, ok := val.(string); ok {  
159 - return v  
160 - }  
161 - }  
162 - return ""  
163 -}  
164 -  
165 -// DefaultString returns the string value for a given key.  
166 -// if err != nil return defaultval  
167 -func (c *JSONConfigContainer) DefaultString(key string, defaultval string) string {  
168 - // TODO FIXME should not use "" to replace non existence  
169 - if v := c.String(key); v != "" {  
170 - return v  
171 - }  
172 - return defaultval  
173 -}  
174 -  
175 -// Strings returns the []string value for a given key.  
176 -func (c *JSONConfigContainer) Strings(key string) []string {  
177 - stringVal := c.String(key)  
178 - if stringVal == "" {  
179 - return nil  
180 - }  
181 - return strings.Split(c.String(key), ";")  
182 -}  
183 -  
184 -// DefaultStrings returns the []string value for a given key.  
185 -// if err != nil return defaultval  
186 -func (c *JSONConfigContainer) DefaultStrings(key string, defaultval []string) []string {  
187 - if v := c.Strings(key); v != nil {  
188 - return v  
189 - }  
190 - return defaultval  
191 -}  
192 -  
193 -// GetSection returns map for the given section  
194 -func (c *JSONConfigContainer) GetSection(section string) (map[string]string, error) {  
195 - if v, ok := c.data[section]; ok {  
196 - return v.(map[string]string), nil  
197 - }  
198 - return nil, errors.New("nonexist section " + section)  
199 -}  
200 -  
201 -// SaveConfigFile save the config into file  
202 -func (c *JSONConfigContainer) SaveConfigFile(filename string) (err error) {  
203 - // Write configuration file by filename.  
204 - f, err := os.Create(filename)  
205 - if err != nil {  
206 - return err  
207 - }  
208 - defer f.Close()  
209 - b, err := json.MarshalIndent(c.data, "", " ")  
210 - if err != nil {  
211 - return err  
212 - }  
213 - _, err = f.Write(b)  
214 - return err  
215 -}  
216 -  
217 -// Set writes a new value for key.  
218 -func (c *JSONConfigContainer) Set(key, val string) error {  
219 - c.Lock()  
220 - defer c.Unlock()  
221 - c.data[key] = val  
222 - return nil  
223 -}  
224 -  
225 -// DIY returns the raw value by a given key.  
226 -func (c *JSONConfigContainer) DIY(key string) (v interface{}, err error) {  
227 - val := c.getData(key)  
228 - if val != nil {  
229 - return val, nil  
230 - }  
231 - return nil, errors.New("not exist key")  
232 -}  
233 -  
234 -// section.key or key  
235 -func (c *JSONConfigContainer) getData(key string) interface{} {  
236 - if len(key) == 0 {  
237 - return nil  
238 - }  
239 -  
240 - c.RLock()  
241 - defer c.RUnlock()  
242 -  
243 - sectionKeys := strings.Split(key, "::")  
244 - if len(sectionKeys) >= 2 {  
245 - curValue, ok := c.data[sectionKeys[0]]  
246 - if !ok {  
247 - return nil  
248 - }  
249 - for _, key := range sectionKeys[1:] {  
250 - if v, ok := curValue.(map[string]interface{}); ok {  
251 - if curValue, ok = v[key]; !ok {  
252 - return nil  
253 - }  
254 - }  
255 - }  
256 - return curValue  
257 - }  
258 - if v, ok := c.data[key]; ok {  
259 - return v  
260 - }  
261 - return nil  
262 -}  
263 -  
264 -func init() {  
265 - Register("json", &JSONConfig{})  
266 -}  
1 -// Copyright 2015 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package context  
16 -  
17 -import (  
18 - "bytes"  
19 - "compress/flate"  
20 - "compress/gzip"  
21 - "compress/zlib"  
22 - "io"  
23 - "net/http"  
24 - "os"  
25 - "strconv"  
26 - "strings"  
27 - "sync"  
28 -)  
29 -  
30 -var (  
31 - //Default size==20B same as nginx  
32 - defaultGzipMinLength = 20  
33 - //Content will only be compressed if content length is either unknown or greater than gzipMinLength.  
34 - gzipMinLength = defaultGzipMinLength  
35 - //The compression level used for deflate compression. (0-9).  
36 - gzipCompressLevel int  
37 - //List of HTTP methods to compress. If not set, only GET requests are compressed.  
38 - includedMethods map[string]bool  
39 - getMethodOnly bool  
40 -)  
41 -  
42 -// InitGzip init the gzipcompress  
43 -func InitGzip(minLength, compressLevel int, methods []string) {  
44 - if minLength >= 0 {  
45 - gzipMinLength = minLength  
46 - }  
47 - gzipCompressLevel = compressLevel  
48 - if gzipCompressLevel < flate.NoCompression || gzipCompressLevel > flate.BestCompression {  
49 - gzipCompressLevel = flate.BestSpeed  
50 - }  
51 - getMethodOnly = (len(methods) == 0) || (len(methods) == 1 && strings.ToUpper(methods[0]) == "GET")  
52 - includedMethods = make(map[string]bool, len(methods))  
53 - for _, v := range methods {  
54 - includedMethods[strings.ToUpper(v)] = true  
55 - }  
56 -}  
57 -  
58 -type resetWriter interface {  
59 - io.Writer  
60 - Reset(w io.Writer)  
61 -}  
62 -  
63 -type nopResetWriter struct {  
64 - io.Writer  
65 -}  
66 -  
67 -func (n nopResetWriter) Reset(w io.Writer) {  
68 - //do nothing  
69 -}  
70 -  
71 -type acceptEncoder struct {  
72 - name string  
73 - levelEncode func(int) resetWriter  
74 - customCompressLevelPool *sync.Pool  
75 - bestCompressionPool *sync.Pool  
76 -}  
77 -  
78 -func (ac acceptEncoder) encode(wr io.Writer, level int) resetWriter {  
79 - if ac.customCompressLevelPool == nil || ac.bestCompressionPool == nil {  
80 - return nopResetWriter{wr}  
81 - }  
82 - var rwr resetWriter  
83 - switch level {  
84 - case flate.BestSpeed:  
85 - rwr = ac.customCompressLevelPool.Get().(resetWriter)  
86 - case flate.BestCompression:  
87 - rwr = ac.bestCompressionPool.Get().(resetWriter)  
88 - default:  
89 - rwr = ac.levelEncode(level)  
90 - }  
91 - rwr.Reset(wr)  
92 - return rwr  
93 -}  
94 -  
95 -func (ac acceptEncoder) put(wr resetWriter, level int) {  
96 - if ac.customCompressLevelPool == nil || ac.bestCompressionPool == nil {  
97 - return  
98 - }  
99 - wr.Reset(nil)  
100 -  
101 - //notice  
102 - //compressionLevel==BestCompression DOES NOT MATTER  
103 - //sync.Pool will not memory leak  
104 -  
105 - switch level {  
106 - case gzipCompressLevel:  
107 - ac.customCompressLevelPool.Put(wr)  
108 - case flate.BestCompression:  
109 - ac.bestCompressionPool.Put(wr)  
110 - }  
111 -}  
112 -  
113 -var (  
114 - noneCompressEncoder = acceptEncoder{"", nil, nil, nil}  
115 - gzipCompressEncoder = acceptEncoder{  
116 - name: "gzip",  
117 - levelEncode: func(level int) resetWriter { wr, _ := gzip.NewWriterLevel(nil, level); return wr },  
118 - customCompressLevelPool: &sync.Pool{New: func() interface{} { wr, _ := gzip.NewWriterLevel(nil, gzipCompressLevel); return wr }},  
119 - bestCompressionPool: &sync.Pool{New: func() interface{} { wr, _ := gzip.NewWriterLevel(nil, flate.BestCompression); return wr }},  
120 - }  
121 -  
122 - //according to the sec :http://tools.ietf.org/html/rfc2616#section-3.5 ,the deflate compress in http is zlib indeed  
123 - //deflate  
124 - //The "zlib" format defined in RFC 1950 [31] in combination with  
125 - //the "deflate" compression mechanism described in RFC 1951 [29].  
126 - deflateCompressEncoder = acceptEncoder{  
127 - name: "deflate",  
128 - levelEncode: func(level int) resetWriter { wr, _ := zlib.NewWriterLevel(nil, level); return wr },  
129 - customCompressLevelPool: &sync.Pool{New: func() interface{} { wr, _ := zlib.NewWriterLevel(nil, gzipCompressLevel); return wr }},  
130 - bestCompressionPool: &sync.Pool{New: func() interface{} { wr, _ := zlib.NewWriterLevel(nil, flate.BestCompression); return wr }},  
131 - }  
132 -)  
133 -  
134 -var (  
135 - encoderMap = map[string]acceptEncoder{ // all the other compress methods will ignore  
136 - "gzip": gzipCompressEncoder,  
137 - "deflate": deflateCompressEncoder,  
138 - "*": gzipCompressEncoder, // * means any compress will accept,we prefer gzip  
139 - "identity": noneCompressEncoder, // identity means none-compress  
140 - }  
141 -)  
142 -  
143 -// WriteFile reads from file and writes to writer by the specific encoding(gzip/deflate)  
144 -func WriteFile(encoding string, writer io.Writer, file *os.File) (bool, string, error) {  
145 - return writeLevel(encoding, writer, file, flate.BestCompression)  
146 -}  
147 -  
148 -// WriteBody reads writes content to writer by the specific encoding(gzip/deflate)  
149 -func WriteBody(encoding string, writer io.Writer, content []byte) (bool, string, error) {  
150 - if encoding == "" || len(content) < gzipMinLength {  
151 - _, err := writer.Write(content)  
152 - return false, "", err  
153 - }  
154 - return writeLevel(encoding, writer, bytes.NewReader(content), gzipCompressLevel)  
155 -}  
156 -  
157 -// writeLevel reads from reader,writes to writer by specific encoding and compress level  
158 -// the compress level is defined by deflate package  
159 -func writeLevel(encoding string, writer io.Writer, reader io.Reader, level int) (bool, string, error) {  
160 - var outputWriter resetWriter  
161 - var err error  
162 - var ce = noneCompressEncoder  
163 -  
164 - if cf, ok := encoderMap[encoding]; ok {  
165 - ce = cf  
166 - }  
167 - encoding = ce.name  
168 - outputWriter = ce.encode(writer, level)  
169 - defer ce.put(outputWriter, level)  
170 -  
171 - _, err = io.Copy(outputWriter, reader)  
172 - if err != nil {  
173 - return false, "", err  
174 - }  
175 -  
176 - switch outputWriter.(type) {  
177 - case io.WriteCloser:  
178 - outputWriter.(io.WriteCloser).Close()  
179 - }  
180 - return encoding != "", encoding, nil  
181 -}  
182 -  
183 -// ParseEncoding will extract the right encoding for response  
184 -// the Accept-Encoding's sec is here:  
185 -// http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.3  
186 -func ParseEncoding(r *http.Request) string {  
187 - if r == nil {  
188 - return ""  
189 - }  
190 - if (getMethodOnly && r.Method == "GET") || includedMethods[r.Method] {  
191 - return parseEncoding(r)  
192 - }  
193 - return ""  
194 -}  
195 -  
196 -type q struct {  
197 - name string  
198 - value float64  
199 -}  
200 -  
201 -func parseEncoding(r *http.Request) string {  
202 - acceptEncoding := r.Header.Get("Accept-Encoding")  
203 - if acceptEncoding == "" {  
204 - return ""  
205 - }  
206 - var lastQ q  
207 - for _, v := range strings.Split(acceptEncoding, ",") {  
208 - v = strings.TrimSpace(v)  
209 - if v == "" {  
210 - continue  
211 - }  
212 - vs := strings.Split(v, ";")  
213 - var cf acceptEncoder  
214 - var ok bool  
215 - if cf, ok = encoderMap[vs[0]]; !ok {  
216 - continue  
217 - }  
218 - if len(vs) == 1 {  
219 - return cf.name  
220 - }  
221 - if len(vs) == 2 {  
222 - f, _ := strconv.ParseFloat(strings.Replace(vs[1], "q=", "", -1), 64)  
223 - if f == 0 {  
224 - continue  
225 - }  
226 - if f > lastQ.value {  
227 - lastQ = q{cf.name, f}  
228 - }  
229 - }  
230 - }  
231 - return lastQ.name  
232 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -// Package context provide the context utils  
16 -// Usage:  
17 -//  
18 -// import "github.com/astaxie/beego/context"  
19 -//  
20 -// ctx := context.Context{Request:req,ResponseWriter:rw}  
21 -//  
22 -// more docs http://beego.me/docs/module/context.md  
23 -package context  
24 -  
25 -import (  
26 - "bufio"  
27 - "crypto/hmac"  
28 - "crypto/sha1"  
29 - "encoding/base64"  
30 - "errors"  
31 - "fmt"  
32 - "net"  
33 - "net/http"  
34 - "strconv"  
35 - "strings"  
36 - "time"  
37 -  
38 - "github.com/astaxie/beego/utils"  
39 -)  
40 -  
41 -// NewContext return the Context with Input and Output  
42 -func NewContext() *Context {  
43 - return &Context{  
44 - Input: NewInput(),  
45 - Output: NewOutput(),  
46 - }  
47 -}  
48 -  
49 -// Context Http request context struct including BeegoInput, BeegoOutput, http.Request and http.ResponseWriter.  
50 -// BeegoInput and BeegoOutput provides some api to operate request and response more easily.  
51 -type Context struct {  
52 - Input *BeegoInput  
53 - Output *BeegoOutput  
54 - Request *http.Request  
55 - ResponseWriter *Response  
56 - _xsrfToken string  
57 -}  
58 -  
59 -// Reset init Context, BeegoInput and BeegoOutput  
60 -func (ctx *Context) Reset(rw http.ResponseWriter, r *http.Request) {  
61 - ctx.Request = r  
62 - if ctx.ResponseWriter == nil {  
63 - ctx.ResponseWriter = &Response{}  
64 - }  
65 - ctx.ResponseWriter.reset(rw)  
66 - ctx.Input.Reset(ctx)  
67 - ctx.Output.Reset(ctx)  
68 - ctx._xsrfToken = ""  
69 -}  
70 -  
71 -// Redirect does redirection to localurl with http header status code.  
72 -func (ctx *Context) Redirect(status int, localurl string) {  
73 - http.Redirect(ctx.ResponseWriter, ctx.Request, localurl, status)  
74 -}  
75 -  
76 -// Abort stops this request.  
77 -// if beego.ErrorMaps exists, panic body.  
78 -func (ctx *Context) Abort(status int, body string) {  
79 - ctx.Output.SetStatus(status)  
80 - panic(body)  
81 -}  
82 -  
83 -// WriteString Write string to response body.  
84 -// it sends response body.  
85 -func (ctx *Context) WriteString(content string) {  
86 - ctx.ResponseWriter.Write([]byte(content))  
87 -}  
88 -  
89 -// GetCookie Get cookie from request by a given key.  
90 -// It's alias of BeegoInput.Cookie.  
91 -func (ctx *Context) GetCookie(key string) string {  
92 - return ctx.Input.Cookie(key)  
93 -}  
94 -  
95 -// SetCookie Set cookie for response.  
96 -// It's alias of BeegoOutput.Cookie.  
97 -func (ctx *Context) SetCookie(name string, value string, others ...interface{}) {  
98 - ctx.Output.Cookie(name, value, others...)  
99 -}  
100 -  
101 -// GetSecureCookie Get secure cookie from request by a given key.  
102 -func (ctx *Context) GetSecureCookie(Secret, key string) (string, bool) {  
103 - val := ctx.Input.Cookie(key)  
104 - if val == "" {  
105 - return "", false  
106 - }  
107 -  
108 - parts := strings.SplitN(val, "|", 3)  
109 -  
110 - if len(parts) != 3 {  
111 - return "", false  
112 - }  
113 -  
114 - vs := parts[0]  
115 - timestamp := parts[1]  
116 - sig := parts[2]  
117 -  
118 - h := hmac.New(sha1.New, []byte(Secret))  
119 - fmt.Fprintf(h, "%s%s", vs, timestamp)  
120 -  
121 - if fmt.Sprintf("%02x", h.Sum(nil)) != sig {  
122 - return "", false  
123 - }  
124 - res, _ := base64.URLEncoding.DecodeString(vs)  
125 - return string(res), true  
126 -}  
127 -  
128 -// SetSecureCookie Set Secure cookie for response.  
129 -func (ctx *Context) SetSecureCookie(Secret, name, value string, others ...interface{}) {  
130 - vs := base64.URLEncoding.EncodeToString([]byte(value))  
131 - timestamp := strconv.FormatInt(time.Now().UnixNano(), 10)  
132 - h := hmac.New(sha1.New, []byte(Secret))  
133 - fmt.Fprintf(h, "%s%s", vs, timestamp)  
134 - sig := fmt.Sprintf("%02x", h.Sum(nil))  
135 - cookie := strings.Join([]string{vs, timestamp, sig}, "|")  
136 - ctx.Output.Cookie(name, cookie, others...)  
137 -}  
138 -  
139 -// XSRFToken creates a xsrf token string and returns.  
140 -func (ctx *Context) XSRFToken(key string, expire int64) string {  
141 - if ctx._xsrfToken == "" {  
142 - token, ok := ctx.GetSecureCookie(key, "_xsrf")  
143 - if !ok {  
144 - token = string(utils.RandomCreateBytes(32))  
145 - ctx.SetSecureCookie(key, "_xsrf", token, expire)  
146 - }  
147 - ctx._xsrfToken = token  
148 - }  
149 - return ctx._xsrfToken  
150 -}  
151 -  
152 -// CheckXSRFCookie checks xsrf token in this request is valid or not.  
153 -// the token can provided in request header "X-Xsrftoken" and "X-CsrfToken"  
154 -// or in form field value named as "_xsrf".  
155 -func (ctx *Context) CheckXSRFCookie() bool {  
156 - token := ctx.Input.Query("_xsrf")  
157 - if token == "" {  
158 - token = ctx.Request.Header.Get("X-Xsrftoken")  
159 - }  
160 - if token == "" {  
161 - token = ctx.Request.Header.Get("X-Csrftoken")  
162 - }  
163 - if token == "" {  
164 - ctx.Abort(403, "'_xsrf' argument missing from POST")  
165 - return false  
166 - }  
167 - if ctx._xsrfToken != token {  
168 - ctx.Abort(403, "XSRF cookie does not match POST argument")  
169 - return false  
170 - }  
171 - return true  
172 -}  
173 -  
174 -// RenderMethodResult renders the return value of a controller method to the output  
175 -func (ctx *Context) RenderMethodResult(result interface{}) {  
176 - if result != nil {  
177 - renderer, ok := result.(Renderer)  
178 - if !ok {  
179 - err, ok := result.(error)  
180 - if ok {  
181 - renderer = errorRenderer(err)  
182 - } else {  
183 - renderer = jsonRenderer(result)  
184 - }  
185 - }  
186 - renderer.Render(ctx)  
187 - }  
188 -}  
189 -  
190 -//Response is a wrapper for the http.ResponseWriter  
191 -//started set to true if response was written to then don't execute other handler  
192 -type Response struct {  
193 - http.ResponseWriter  
194 - Started bool  
195 - Status int  
196 -}  
197 -  
198 -func (r *Response) reset(rw http.ResponseWriter) {  
199 - r.ResponseWriter = rw  
200 - r.Status = 0  
201 - r.Started = false  
202 -}  
203 -  
204 -// Write writes the data to the connection as part of an HTTP reply,  
205 -// and sets `started` to true.  
206 -// started means the response has sent out.  
207 -func (r *Response) Write(p []byte) (int, error) {  
208 - r.Started = true  
209 - return r.ResponseWriter.Write(p)  
210 -}  
211 -  
212 -// WriteHeader sends an HTTP response header with status code,  
213 -// and sets `started` to true.  
214 -func (r *Response) WriteHeader(code int) {  
215 - if r.Status > 0 {  
216 - //prevent multiple response.WriteHeader calls  
217 - return  
218 - }  
219 - r.Status = code  
220 - r.Started = true  
221 - r.ResponseWriter.WriteHeader(code)  
222 -}  
223 -  
224 -// Hijack hijacker for http  
225 -func (r *Response) Hijack() (net.Conn, *bufio.ReadWriter, error) {  
226 - hj, ok := r.ResponseWriter.(http.Hijacker)  
227 - if !ok {  
228 - return nil, nil, errors.New("webserver doesn't support hijacking")  
229 - }  
230 - return hj.Hijack()  
231 -}  
232 -  
233 -// Flush http.Flusher  
234 -func (r *Response) Flush() {  
235 - if f, ok := r.ResponseWriter.(http.Flusher); ok {  
236 - f.Flush()  
237 - }  
238 -}  
239 -  
240 -// CloseNotify http.CloseNotifier  
241 -func (r *Response) CloseNotify() <-chan bool {  
242 - if cn, ok := r.ResponseWriter.(http.CloseNotifier); ok {  
243 - return cn.CloseNotify()  
244 - }  
245 - return nil  
246 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package context  
16 -  
17 -import (  
18 - "bytes"  
19 - "compress/gzip"  
20 - "errors"  
21 - "io"  
22 - "io/ioutil"  
23 - "net"  
24 - "net/http"  
25 - "net/url"  
26 - "reflect"  
27 - "regexp"  
28 - "strconv"  
29 - "strings"  
30 -  
31 - "github.com/astaxie/beego/session"  
32 -)  
33 -  
34 -// Regexes for checking the accept headers  
35 -// TODO make sure these are correct  
36 -var (  
37 - acceptsHTMLRegex = regexp.MustCompile(`(text/html|application/xhtml\+xml)(?:,|$)`)  
38 - acceptsXMLRegex = regexp.MustCompile(`(application/xml|text/xml)(?:,|$)`)  
39 - acceptsJSONRegex = regexp.MustCompile(`(application/json)(?:,|$)`)  
40 - acceptsYAMLRegex = regexp.MustCompile(`(application/x-yaml)(?:,|$)`)  
41 - maxParam = 50  
42 -)  
43 -  
44 -// BeegoInput operates the http request header, data, cookie and body.  
45 -// it also contains router params and current session.  
46 -type BeegoInput struct {  
47 - Context *Context  
48 - CruSession session.Store  
49 - pnames []string  
50 - pvalues []string  
51 - data map[interface{}]interface{} // store some values in this context when calling context in filter or controller.  
52 - RequestBody []byte  
53 - RunMethod string  
54 - RunController reflect.Type  
55 -}  
56 -  
57 -// NewInput return BeegoInput generated by Context.  
58 -func NewInput() *BeegoInput {  
59 - return &BeegoInput{  
60 - pnames: make([]string, 0, maxParam),  
61 - pvalues: make([]string, 0, maxParam),  
62 - data: make(map[interface{}]interface{}),  
63 - }  
64 -}  
65 -  
66 -// Reset init the BeegoInput  
67 -func (input *BeegoInput) Reset(ctx *Context) {  
68 - input.Context = ctx  
69 - input.CruSession = nil  
70 - input.pnames = input.pnames[:0]  
71 - input.pvalues = input.pvalues[:0]  
72 - input.data = nil  
73 - input.RequestBody = []byte{}  
74 -}  
75 -  
76 -// Protocol returns request protocol name, such as HTTP/1.1 .  
77 -func (input *BeegoInput) Protocol() string {  
78 - return input.Context.Request.Proto  
79 -}  
80 -  
81 -// URI returns full request url with query string, fragment.  
82 -func (input *BeegoInput) URI() string {  
83 - return input.Context.Request.RequestURI  
84 -}  
85 -  
86 -// URL returns request url path (without query string, fragment).  
87 -func (input *BeegoInput) URL() string {  
88 - return input.Context.Request.URL.Path  
89 -}  
90 -  
91 -// Site returns base site url as scheme://domain type.  
92 -func (input *BeegoInput) Site() string {  
93 - return input.Scheme() + "://" + input.Domain()  
94 -}  
95 -  
96 -// Scheme returns request scheme as "http" or "https".  
97 -func (input *BeegoInput) Scheme() string {  
98 - if scheme := input.Header("X-Forwarded-Proto"); scheme != "" {  
99 - return scheme  
100 - }  
101 - if input.Context.Request.URL.Scheme != "" {  
102 - return input.Context.Request.URL.Scheme  
103 - }  
104 - if input.Context.Request.TLS == nil {  
105 - return "http"  
106 - }  
107 - return "https"  
108 -}  
109 -  
110 -// Domain returns host name.  
111 -// Alias of Host method.  
112 -func (input *BeegoInput) Domain() string {  
113 - return input.Host()  
114 -}  
115 -  
116 -// Host returns host name.  
117 -// if no host info in request, return localhost.  
118 -func (input *BeegoInput) Host() string {  
119 - if input.Context.Request.Host != "" {  
120 - if hostPart, _, err := net.SplitHostPort(input.Context.Request.Host); err == nil {  
121 - return hostPart  
122 - }  
123 - return input.Context.Request.Host  
124 - }  
125 - return "localhost"  
126 -}  
127 -  
128 -// Method returns http request method.  
129 -func (input *BeegoInput) Method() string {  
130 - return input.Context.Request.Method  
131 -}  
132 -  
133 -// Is returns boolean of this request is on given method, such as Is("POST").  
134 -func (input *BeegoInput) Is(method string) bool {  
135 - return input.Method() == method  
136 -}  
137 -  
138 -// IsGet Is this a GET method request?  
139 -func (input *BeegoInput) IsGet() bool {  
140 - return input.Is("GET")  
141 -}  
142 -  
143 -// IsPost Is this a POST method request?  
144 -func (input *BeegoInput) IsPost() bool {  
145 - return input.Is("POST")  
146 -}  
147 -  
148 -// IsHead Is this a Head method request?  
149 -func (input *BeegoInput) IsHead() bool {  
150 - return input.Is("HEAD")  
151 -}  
152 -  
153 -// IsOptions Is this a OPTIONS method request?  
154 -func (input *BeegoInput) IsOptions() bool {  
155 - return input.Is("OPTIONS")  
156 -}  
157 -  
158 -// IsPut Is this a PUT method request?  
159 -func (input *BeegoInput) IsPut() bool {  
160 - return input.Is("PUT")  
161 -}  
162 -  
163 -// IsDelete Is this a DELETE method request?  
164 -func (input *BeegoInput) IsDelete() bool {  
165 - return input.Is("DELETE")  
166 -}  
167 -  
168 -// IsPatch Is this a PATCH method request?  
169 -func (input *BeegoInput) IsPatch() bool {  
170 - return input.Is("PATCH")  
171 -}  
172 -  
173 -// IsAjax returns boolean of this request is generated by ajax.  
174 -func (input *BeegoInput) IsAjax() bool {  
175 - return input.Header("X-Requested-With") == "XMLHttpRequest"  
176 -}  
177 -  
178 -// IsSecure returns boolean of this request is in https.  
179 -func (input *BeegoInput) IsSecure() bool {  
180 - return input.Scheme() == "https"  
181 -}  
182 -  
183 -// IsWebsocket returns boolean of this request is in webSocket.  
184 -func (input *BeegoInput) IsWebsocket() bool {  
185 - return input.Header("Upgrade") == "websocket"  
186 -}  
187 -  
188 -// IsUpload returns boolean of whether file uploads in this request or not..  
189 -func (input *BeegoInput) IsUpload() bool {  
190 - return strings.Contains(input.Header("Content-Type"), "multipart/form-data")  
191 -}  
192 -  
193 -// AcceptsHTML Checks if request accepts html response  
194 -func (input *BeegoInput) AcceptsHTML() bool {  
195 - return acceptsHTMLRegex.MatchString(input.Header("Accept"))  
196 -}  
197 -  
198 -// AcceptsXML Checks if request accepts xml response  
199 -func (input *BeegoInput) AcceptsXML() bool {  
200 - return acceptsXMLRegex.MatchString(input.Header("Accept"))  
201 -}  
202 -  
203 -// AcceptsJSON Checks if request accepts json response  
204 -func (input *BeegoInput) AcceptsJSON() bool {  
205 - return acceptsJSONRegex.MatchString(input.Header("Accept"))  
206 -}  
207 -// AcceptsYAML Checks if request accepts json response  
208 -func (input *BeegoInput) AcceptsYAML() bool {  
209 - return acceptsYAMLRegex.MatchString(input.Header("Accept"))  
210 -}  
211 -  
212 -// IP returns request client ip.  
213 -// if in proxy, return first proxy id.  
214 -// if error, return RemoteAddr.  
215 -func (input *BeegoInput) IP() string {  
216 - ips := input.Proxy()  
217 - if len(ips) > 0 && ips[0] != "" {  
218 - rip, _, err := net.SplitHostPort(ips[0])  
219 - if err != nil {  
220 - rip = ips[0]  
221 - }  
222 - return rip  
223 - }  
224 - if ip, _, err := net.SplitHostPort(input.Context.Request.RemoteAddr); err == nil {  
225 - return ip  
226 - }  
227 - return input.Context.Request.RemoteAddr  
228 -}  
229 -  
230 -// Proxy returns proxy client ips slice.  
231 -func (input *BeegoInput) Proxy() []string {  
232 - if ips := input.Header("X-Forwarded-For"); ips != "" {  
233 - return strings.Split(ips, ",")  
234 - }  
235 - return []string{}  
236 -}  
237 -  
238 -// Referer returns http referer header.  
239 -func (input *BeegoInput) Referer() string {  
240 - return input.Header("Referer")  
241 -}  
242 -  
243 -// Refer returns http referer header.  
244 -func (input *BeegoInput) Refer() string {  
245 - return input.Referer()  
246 -}  
247 -  
248 -// SubDomains returns sub domain string.  
249 -// if aa.bb.domain.com, returns aa.bb .  
250 -func (input *BeegoInput) SubDomains() string {  
251 - parts := strings.Split(input.Host(), ".")  
252 - if len(parts) >= 3 {  
253 - return strings.Join(parts[:len(parts)-2], ".")  
254 - }  
255 - return ""  
256 -}  
257 -  
258 -// Port returns request client port.  
259 -// when error or empty, return 80.  
260 -func (input *BeegoInput) Port() int {  
261 - if _, portPart, err := net.SplitHostPort(input.Context.Request.Host); err == nil {  
262 - port, _ := strconv.Atoi(portPart)  
263 - return port  
264 - }  
265 - return 80  
266 -}  
267 -  
268 -// UserAgent returns request client user agent string.  
269 -func (input *BeegoInput) UserAgent() string {  
270 - return input.Header("User-Agent")  
271 -}  
272 -  
273 -// ParamsLen return the length of the params  
274 -func (input *BeegoInput) ParamsLen() int {  
275 - return len(input.pnames)  
276 -}  
277 -  
278 -// Param returns router param by a given key.  
279 -func (input *BeegoInput) Param(key string) string {  
280 - for i, v := range input.pnames {  
281 - if v == key && i <= len(input.pvalues) {  
282 - return input.pvalues[i]  
283 - }  
284 - }  
285 - return ""  
286 -}  
287 -  
288 -// Params returns the map[key]value.  
289 -func (input *BeegoInput) Params() map[string]string {  
290 - m := make(map[string]string)  
291 - for i, v := range input.pnames {  
292 - if i <= len(input.pvalues) {  
293 - m[v] = input.pvalues[i]  
294 - }  
295 - }  
296 - return m  
297 -}  
298 -  
299 -// SetParam will set the param with key and value  
300 -func (input *BeegoInput) SetParam(key, val string) {  
301 - // check if already exists  
302 - for i, v := range input.pnames {  
303 - if v == key && i <= len(input.pvalues) {  
304 - input.pvalues[i] = val  
305 - return  
306 - }  
307 - }  
308 - input.pvalues = append(input.pvalues, val)  
309 - input.pnames = append(input.pnames, key)  
310 -}  
311 -  
312 -// ResetParams clears any of the input's Params  
313 -// This function is used to clear parameters so they may be reset between filter  
314 -// passes.  
315 -func (input *BeegoInput) ResetParams() {  
316 - input.pnames = input.pnames[:0]  
317 - input.pvalues = input.pvalues[:0]  
318 -}  
319 -  
320 -// Query returns input data item string by a given string.  
321 -func (input *BeegoInput) Query(key string) string {  
322 - if val := input.Param(key); val != "" {  
323 - return val  
324 - }  
325 - if input.Context.Request.Form == nil {  
326 - input.Context.Request.ParseForm()  
327 - }  
328 - return input.Context.Request.Form.Get(key)  
329 -}  
330 -  
331 -// Header returns request header item string by a given string.  
332 -// if non-existed, return empty string.  
333 -func (input *BeegoInput) Header(key string) string {  
334 - return input.Context.Request.Header.Get(key)  
335 -}  
336 -  
337 -// Cookie returns request cookie item string by a given key.  
338 -// if non-existed, return empty string.  
339 -func (input *BeegoInput) Cookie(key string) string {  
340 - ck, err := input.Context.Request.Cookie(key)  
341 - if err != nil {  
342 - return ""  
343 - }  
344 - return ck.Value  
345 -}  
346 -  
347 -// Session returns current session item value by a given key.  
348 -// if non-existed, return nil.  
349 -func (input *BeegoInput) Session(key interface{}) interface{} {  
350 - return input.CruSession.Get(key)  
351 -}  
352 -  
353 -// CopyBody returns the raw request body data as bytes.  
354 -func (input *BeegoInput) CopyBody(MaxMemory int64) []byte {  
355 - if input.Context.Request.Body == nil {  
356 - return []byte{}  
357 - }  
358 -  
359 - var requestbody []byte  
360 - safe := &io.LimitedReader{R: input.Context.Request.Body, N: MaxMemory}  
361 - if input.Header("Content-Encoding") == "gzip" {  
362 - reader, err := gzip.NewReader(safe)  
363 - if err != nil {  
364 - return nil  
365 - }  
366 - requestbody, _ = ioutil.ReadAll(reader)  
367 - } else {  
368 - requestbody, _ = ioutil.ReadAll(safe)  
369 - }  
370 -  
371 - input.Context.Request.Body.Close()  
372 - bf := bytes.NewBuffer(requestbody)  
373 - input.Context.Request.Body = http.MaxBytesReader(input.Context.ResponseWriter, ioutil.NopCloser(bf), MaxMemory)  
374 - input.RequestBody = requestbody  
375 - return requestbody  
376 -}  
377 -  
378 -// Data return the implicit data in the input  
379 -func (input *BeegoInput) Data() map[interface{}]interface{} {  
380 - if input.data == nil {  
381 - input.data = make(map[interface{}]interface{})  
382 - }  
383 - return input.data  
384 -}  
385 -  
386 -// GetData returns the stored data in this context.  
387 -func (input *BeegoInput) GetData(key interface{}) interface{} {  
388 - if v, ok := input.data[key]; ok {  
389 - return v  
390 - }  
391 - return nil  
392 -}  
393 -  
394 -// SetData stores data with given key in this context.  
395 -// This data are only available in this context.  
396 -func (input *BeegoInput) SetData(key, val interface{}) {  
397 - if input.data == nil {  
398 - input.data = make(map[interface{}]interface{})  
399 - }  
400 - input.data[key] = val  
401 -}  
402 -  
403 -// ParseFormOrMulitForm parseForm or parseMultiForm based on Content-type  
404 -func (input *BeegoInput) ParseFormOrMulitForm(maxMemory int64) error {  
405 - // Parse the body depending on the content type.  
406 - if strings.Contains(input.Header("Content-Type"), "multipart/form-data") {  
407 - if err := input.Context.Request.ParseMultipartForm(maxMemory); err != nil {  
408 - return errors.New("Error parsing request body:" + err.Error())  
409 - }  
410 - } else if err := input.Context.Request.ParseForm(); err != nil {  
411 - return errors.New("Error parsing request body:" + err.Error())  
412 - }  
413 - return nil  
414 -}  
415 -  
416 -// Bind data from request.Form[key] to dest  
417 -// like /?id=123&isok=true&ft=1.2&ol[0]=1&ol[1]=2&ul[]=str&ul[]=array&user.Name=astaxie  
418 -// var id int beegoInput.Bind(&id, "id") id ==123  
419 -// var isok bool beegoInput.Bind(&isok, "isok") isok ==true  
420 -// var ft float64 beegoInput.Bind(&ft, "ft") ft ==1.2  
421 -// ol := make([]int, 0, 2) beegoInput.Bind(&ol, "ol") ol ==[1 2]  
422 -// ul := make([]string, 0, 2) beegoInput.Bind(&ul, "ul") ul ==[str array]  
423 -// user struct{Name} beegoInput.Bind(&user, "user") user == {Name:"astaxie"}  
424 -func (input *BeegoInput) Bind(dest interface{}, key string) error {  
425 - value := reflect.ValueOf(dest)  
426 - if value.Kind() != reflect.Ptr {  
427 - return errors.New("beego: non-pointer passed to Bind: " + key)  
428 - }  
429 - value = value.Elem()  
430 - if !value.CanSet() {  
431 - return errors.New("beego: non-settable variable passed to Bind: " + key)  
432 - }  
433 - typ := value.Type()  
434 - // Get real type if dest define with interface{}.  
435 - // e.g var dest interface{} dest=1.0  
436 - if value.Kind() == reflect.Interface {  
437 - typ = value.Elem().Type()  
438 - }  
439 - rv := input.bind(key, typ)  
440 - if !rv.IsValid() {  
441 - return errors.New("beego: reflect value is empty")  
442 - }  
443 - value.Set(rv)  
444 - return nil  
445 -}  
446 -  
447 -func (input *BeegoInput) bind(key string, typ reflect.Type) reflect.Value {  
448 - if input.Context.Request.Form == nil {  
449 - input.Context.Request.ParseForm()  
450 - }  
451 - rv := reflect.Zero(typ)  
452 - switch typ.Kind() {  
453 - case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:  
454 - val := input.Query(key)  
455 - if len(val) == 0 {  
456 - return rv  
457 - }  
458 - rv = input.bindInt(val, typ)  
459 - case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:  
460 - val := input.Query(key)  
461 - if len(val) == 0 {  
462 - return rv  
463 - }  
464 - rv = input.bindUint(val, typ)  
465 - case reflect.Float32, reflect.Float64:  
466 - val := input.Query(key)  
467 - if len(val) == 0 {  
468 - return rv  
469 - }  
470 - rv = input.bindFloat(val, typ)  
471 - case reflect.String:  
472 - val := input.Query(key)  
473 - if len(val) == 0 {  
474 - return rv  
475 - }  
476 - rv = input.bindString(val, typ)  
477 - case reflect.Bool:  
478 - val := input.Query(key)  
479 - if len(val) == 0 {  
480 - return rv  
481 - }  
482 - rv = input.bindBool(val, typ)  
483 - case reflect.Slice:  
484 - rv = input.bindSlice(&input.Context.Request.Form, key, typ)  
485 - case reflect.Struct:  
486 - rv = input.bindStruct(&input.Context.Request.Form, key, typ)  
487 - case reflect.Ptr:  
488 - rv = input.bindPoint(key, typ)  
489 - case reflect.Map:  
490 - rv = input.bindMap(&input.Context.Request.Form, key, typ)  
491 - }  
492 - return rv  
493 -}  
494 -  
495 -func (input *BeegoInput) bindValue(val string, typ reflect.Type) reflect.Value {  
496 - rv := reflect.Zero(typ)  
497 - switch typ.Kind() {  
498 - case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:  
499 - rv = input.bindInt(val, typ)  
500 - case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:  
501 - rv = input.bindUint(val, typ)  
502 - case reflect.Float32, reflect.Float64:  
503 - rv = input.bindFloat(val, typ)  
504 - case reflect.String:  
505 - rv = input.bindString(val, typ)  
506 - case reflect.Bool:  
507 - rv = input.bindBool(val, typ)  
508 - case reflect.Slice:  
509 - rv = input.bindSlice(&url.Values{"": {val}}, "", typ)  
510 - case reflect.Struct:  
511 - rv = input.bindStruct(&url.Values{"": {val}}, "", typ)  
512 - case reflect.Ptr:  
513 - rv = input.bindPoint(val, typ)  
514 - case reflect.Map:  
515 - rv = input.bindMap(&url.Values{"": {val}}, "", typ)  
516 - }  
517 - return rv  
518 -}  
519 -  
520 -func (input *BeegoInput) bindInt(val string, typ reflect.Type) reflect.Value {  
521 - intValue, err := strconv.ParseInt(val, 10, 64)  
522 - if err != nil {  
523 - return reflect.Zero(typ)  
524 - }  
525 - pValue := reflect.New(typ)  
526 - pValue.Elem().SetInt(intValue)  
527 - return pValue.Elem()  
528 -}  
529 -  
530 -func (input *BeegoInput) bindUint(val string, typ reflect.Type) reflect.Value {  
531 - uintValue, err := strconv.ParseUint(val, 10, 64)  
532 - if err != nil {  
533 - return reflect.Zero(typ)  
534 - }  
535 - pValue := reflect.New(typ)  
536 - pValue.Elem().SetUint(uintValue)  
537 - return pValue.Elem()  
538 -}  
539 -  
540 -func (input *BeegoInput) bindFloat(val string, typ reflect.Type) reflect.Value {  
541 - floatValue, err := strconv.ParseFloat(val, 64)  
542 - if err != nil {  
543 - return reflect.Zero(typ)  
544 - }  
545 - pValue := reflect.New(typ)  
546 - pValue.Elem().SetFloat(floatValue)  
547 - return pValue.Elem()  
548 -}  
549 -  
550 -func (input *BeegoInput) bindString(val string, typ reflect.Type) reflect.Value {  
551 - return reflect.ValueOf(val)  
552 -}  
553 -  
554 -func (input *BeegoInput) bindBool(val string, typ reflect.Type) reflect.Value {  
555 - val = strings.TrimSpace(strings.ToLower(val))  
556 - switch val {  
557 - case "true", "on", "1":  
558 - return reflect.ValueOf(true)  
559 - }  
560 - return reflect.ValueOf(false)  
561 -}  
562 -  
563 -type sliceValue struct {  
564 - index int // Index extracted from brackets. If -1, no index was provided.  
565 - value reflect.Value // the bound value for this slice element.  
566 -}  
567 -  
568 -func (input *BeegoInput) bindSlice(params *url.Values, key string, typ reflect.Type) reflect.Value {  
569 - maxIndex := -1  
570 - numNoIndex := 0  
571 - sliceValues := []sliceValue{}  
572 - for reqKey, vals := range *params {  
573 - if !strings.HasPrefix(reqKey, key+"[") {  
574 - continue  
575 - }  
576 - // Extract the index, and the index where a sub-key starts. (e.g. field[0].subkey)  
577 - index := -1  
578 - leftBracket, rightBracket := len(key), strings.Index(reqKey[len(key):], "]")+len(key)  
579 - if rightBracket > leftBracket+1 {  
580 - index, _ = strconv.Atoi(reqKey[leftBracket+1 : rightBracket])  
581 - }  
582 - subKeyIndex := rightBracket + 1  
583 -  
584 - // Handle the indexed case.  
585 - if index > -1 {  
586 - if index > maxIndex {  
587 - maxIndex = index  
588 - }  
589 - sliceValues = append(sliceValues, sliceValue{  
590 - index: index,  
591 - value: input.bind(reqKey[:subKeyIndex], typ.Elem()),  
592 - })  
593 - continue  
594 - }  
595 -  
596 - // It's an un-indexed element. (e.g. element[])  
597 - numNoIndex += len(vals)  
598 - for _, val := range vals {  
599 - // Unindexed values can only be direct-bound.  
600 - sliceValues = append(sliceValues, sliceValue{  
601 - index: -1,  
602 - value: input.bindValue(val, typ.Elem()),  
603 - })  
604 - }  
605 - }  
606 - resultArray := reflect.MakeSlice(typ, maxIndex+1, maxIndex+1+numNoIndex)  
607 - for _, sv := range sliceValues {  
608 - if sv.index != -1 {  
609 - resultArray.Index(sv.index).Set(sv.value)  
610 - } else {  
611 - resultArray = reflect.Append(resultArray, sv.value)  
612 - }  
613 - }  
614 - return resultArray  
615 -}  
616 -  
617 -func (input *BeegoInput) bindStruct(params *url.Values, key string, typ reflect.Type) reflect.Value {  
618 - result := reflect.New(typ).Elem()  
619 - fieldValues := make(map[string]reflect.Value)  
620 - for reqKey, val := range *params {  
621 - var fieldName string  
622 - if strings.HasPrefix(reqKey, key+".") {  
623 - fieldName = reqKey[len(key)+1:]  
624 - } else if strings.HasPrefix(reqKey, key+"[") && reqKey[len(reqKey)-1] == ']' {  
625 - fieldName = reqKey[len(key)+1 : len(reqKey)-1]  
626 - } else {  
627 - continue  
628 - }  
629 -  
630 - if _, ok := fieldValues[fieldName]; !ok {  
631 - // Time to bind this field. Get it and make sure we can set it.  
632 - fieldValue := result.FieldByName(fieldName)  
633 - if !fieldValue.IsValid() {  
634 - continue  
635 - }  
636 - if !fieldValue.CanSet() {  
637 - continue  
638 - }  
639 - boundVal := input.bindValue(val[0], fieldValue.Type())  
640 - fieldValue.Set(boundVal)  
641 - fieldValues[fieldName] = boundVal  
642 - }  
643 - }  
644 -  
645 - return result  
646 -}  
647 -  
648 -func (input *BeegoInput) bindPoint(key string, typ reflect.Type) reflect.Value {  
649 - return input.bind(key, typ.Elem()).Addr()  
650 -}  
651 -  
652 -func (input *BeegoInput) bindMap(params *url.Values, key string, typ reflect.Type) reflect.Value {  
653 - var (  
654 - result = reflect.MakeMap(typ)  
655 - keyType = typ.Key()  
656 - valueType = typ.Elem()  
657 - )  
658 - for paramName, values := range *params {  
659 - if !strings.HasPrefix(paramName, key+"[") || paramName[len(paramName)-1] != ']' {  
660 - continue  
661 - }  
662 -  
663 - key := paramName[len(key)+1 : len(paramName)-1]  
664 - result.SetMapIndex(input.bindValue(key, keyType), input.bindValue(values[0], valueType))  
665 - }  
666 - return result  
667 -}  
1 -// Copyright 2014 beego Author. All Rights Reserved.  
2 -//  
3 -// Licensed under the Apache License, Version 2.0 (the "License");  
4 -// you may not use this file except in compliance with the License.  
5 -// You may obtain a copy of the License at  
6 -//  
7 -// http://www.apache.org/licenses/LICENSE-2.0  
8 -//  
9 -// Unless required by applicable law or agreed to in writing, software  
10 -// distributed under the License is distributed on an "AS IS" BASIS,  
11 -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  
12 -// See the License for the specific language governing permissions and  
13 -// limitations under the License.  
14 -  
15 -package context  
16 -  
17 -import (  
18 - "bytes"  
19 - "encoding/json"  
20 - "encoding/xml"  
21 - "errors"  
22 - "fmt"  
23 - "html/template"  
24 - "io"  
25 - "mime"  
26 - "net/http"  
27 - "net/url"  
28 - "os"  
29 - "path/filepath"  
30 - "strconv"  
31 - "strings"  
32 - "time"  
33 - "gopkg.in/yaml.v2"  
34 -)  
35 -  
36 -// BeegoOutput does work for sending response header.  
37 -type BeegoOutput struct {  
38 - Context *Context  
39 - Status int  
40 - EnableGzip bool  
41 -}  
42 -  
43 -// NewOutput returns new BeegoOutput.  
44 -// it contains nothing now.  
45 -func NewOutput() *BeegoOutput {  
46 - return &BeegoOutput{}  
47 -}  
48 -  
49 -// Reset init BeegoOutput  
50 -func (output *BeegoOutput) Reset(ctx *Context) {  
51 - output.Context = ctx  
52 - output.Status = 0  
53 -}  
54 -  
55 -// Header sets response header item string via given key.  
56 -func (output *BeegoOutput) Header(key, val string) {  
57 - output.Context.ResponseWriter.Header().Set(key, val)  
58 -}  
59 -  
60 -// Body sets response body content.  
61 -// if EnableGzip, compress content string.  
62 -// it sends out response body directly.  
63 -func (output *BeegoOutput) Body(content []byte) error {  
64 - var encoding string  
65 - var buf = &bytes.Buffer{}  
66 - if output.EnableGzip {  
67 - encoding = ParseEncoding(output.Context.Request)  
68 - }  
69 - if b, n, _ := WriteBody(encoding, buf, content); b {  
70 - output.Header("Content-Encoding", n)  
71 - output.Header("Content-Length", strconv.Itoa(buf.Len()))  
72 - } else {  
73 - output.Header("Content-Length", strconv.Itoa(len(content)))  
74 - }  
75 - // Write status code if it has been set manually  
76 - // Set it to 0 afterwards to prevent "multiple response.WriteHeader calls"  
77 - if output.Status != 0 {  
78 - output.Context.ResponseWriter.WriteHeader(output.Status)  
79 - output.Status = 0  
80 - } else {  
81 - output.Context.ResponseWriter.Started = true  
82 - }  
83 - io.Copy(output.Context.ResponseWriter, buf)  
84 - return nil  
85 -}  
86 -  
87 -// Cookie sets cookie value via given key.  
88 -// others are ordered as cookie's max age time, path,domain, secure and httponly.  
89 -func (output *BeegoOutput) Cookie(name string, value string, others ...interface{}) {  
90 - var b bytes.Buffer  
91 - fmt.Fprintf(&b, "%s=%s", sanitizeName(name), sanitizeValue(value))  
92 -  
93 - //fix cookie not work in IE  
94 - if len(others) > 0 {  
95 - var maxAge int64  
96 -  
97 - switch v := others[0].(type) {  
98 - case int:  
99 - maxAge = int64(v)  
100 - case int32:  
101 - maxAge = int64(v)  
102 - case int64:  
103 - maxAge = v  
104 - }  
105 -  
106 - switch {  
107 - case maxAge > 0:  
108 - fmt.Fprintf(&b, "; Expires=%s; Max-Age=%d", time.Now().Add(time.Duration(maxAge)*time.Second).UTC().Format(time.RFC1123), maxAge)  
109 - case maxAge < 0:  
110 - fmt.Fprintf(&b, "; Max-Age=0")  
111 - }  
112 - }  
113 -  
114 - // the settings below  
115 - // Path, Domain, Secure, HttpOnly  
116 - // can use nil skip set  
117 -  
118 - // default "/"  
119 - if len(others) > 1 {  
120 - if v, ok := others[1].(string); ok && len(v) > 0 {  
121 - fmt.Fprintf(&b, "; Path=%s", sanitizeValue(v))  
122 - }  
123 - } else {  
124 - fmt.Fprintf(&b, "; Path=%s", "/")  
125 - }  
126 -  
127 - // default empty  
128 - if len(others) > 2 {  
129 - if v, ok := others[2].(string); ok && len(v) > 0 {  
130 - fmt.Fprintf(&b, "; Domain=%s", sanitizeValue(v))  
131 - }  
132 - }  
133 -  
134 - // default empty  
135 - if len(others) > 3 {  
136 - var secure bool  
137 - switch v := others[3].(type) {  
138 - case bool:  
139 - secure = v  
140 - default:  
141 - if others[3] != nil {  
142 - secure = true  
143 - }  
144 - }  
145 - if secure {  
146 - fmt.Fprintf(&b, "; Secure")  
147 - }  
148 - }  
149 -  
150 - // default false. for session cookie default true  
151 - if len(others) > 4 {  
152 - if v, ok := others[4].(bool); ok && v {  
153 - fmt.Fprintf(&b, "; HttpOnly")  
154 - }  
155 - }  
156 -  
157 - output.Context.ResponseWriter.Header().Add("Set-Cookie", b.String())  
158 -}  
159 -  
160 -var cookieNameSanitizer = strings.NewReplacer("\n", "-", "\r", "-")  
161 -  
162 -func sanitizeName(n string) string {  
163 - return cookieNameSanitizer.Replace(n)  
164 -}  
165 -  
166 -var cookieValueSanitizer = strings.NewReplacer("\n", " ", "\r", " ", ";", " ")  
167 -  
168 -func sanitizeValue(v string) string {  
169 - return cookieValueSanitizer.Replace(v)  
170 -}  
171 -  
172 -func jsonRenderer(value interface{}) Renderer {  
173 - return rendererFunc(func(ctx *Context) {  
174 - ctx.Output.JSON(value, false, false)  
175 - })  
176 -}  
177 -  
178 -func errorRenderer(err error) Renderer {  
179 - return rendererFunc(func(ctx *Context) {  
180 - ctx.Output.SetStatus(500)  
181 - ctx.Output.Body([]byte(err.Error()))  
182 - })  
183 -}  
184 -  
185 -// JSON writes json to response body.  
186 -// if encoding is true, it converts utf-8 to \u0000 type.  
187 -func (output *BeegoOutput) JSON(data interface{}, hasIndent bool, encoding bool) error {  
188 - output.Header("Content-Type", "application/json; charset=utf-8")  
189 - var content []byte  
190 - var err error  
191 - if hasIndent {  
192 - content, err = json.MarshalIndent(data, "", " ")  
193 - } else {  
194 - content, err = json.Marshal(data)  
195 - }  
196 - if err != nil {  
197 - http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)  
198 - return err  
199 - }  
200 - if encoding {  
201 - content = []byte(stringsToJSON(string(content)))  
202 - }  
203 - return output.Body(content)  
204 -}  
205 -  
206 -  
207 -// YAML writes yaml to response body.  
208 -func (output *BeegoOutput) YAML(data interface{}) error {  
209 - output.Header("Content-Type", "application/application/x-yaml; charset=utf-8")  
210 - var content []byte  
211 - var err error  
212 - content, err = yaml.Marshal(data)  
213 - if err != nil {  
214 - http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)  
215 - return err  
216 - }  
217 - return output.Body(content)  
218 -}  
219 -  
220 -// JSONP writes jsonp to response body.  
221 -func (output *BeegoOutput) JSONP(data interface{}, hasIndent bool) error {  
222 - output.Header("Content-Type", "application/javascript; charset=utf-8")  
223 - var content []byte  
224 - var err error  
225 - if hasIndent {  
226 - content, err = json.MarshalIndent(data, "", " ")  
227 - } else {  
228 - content, err = json.Marshal(data)  
229 - }  
230 - if err != nil {  
231 - http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)  
232 - return err  
233 - }  
234 - callback := output.Context.Input.Query("callback")  
235 - if callback == "" {  
236 - return errors.New(`"callback" parameter required`)  
237 - }  
238 - callback = template.JSEscapeString(callback)  
239 - callbackContent := bytes.NewBufferString(" if(window." + callback + ")" + callback)  
240 - callbackContent.WriteString("(")  
241 - callbackContent.Write(content)  
242 - callbackContent.WriteString(");\r\n")  
243 - return output.Body(callbackContent.Bytes())  
244 -}  
245 -  
246 -// XML writes xml string to response body.  
247 -func (output *BeegoOutput) XML(data interface{}, hasIndent bool) error {  
248 - output.Header("Content-Type", "application/xml; charset=utf-8")  
249 - var content []byte  
250 - var err error  
251 - if hasIndent {  
252 - content, err = xml.MarshalIndent(data, "", " ")  
253 - } else {  
254 - content, err = xml.Marshal(data)  
255 - }  
256 - if err != nil {  
257 - http.Error(output.Context.ResponseWriter, err.Error(), http.StatusInternalServerError)  
258 - return err  
259 - }  
260 - return output.Body(content)  
261 -}  
262 -  
263 -// Download forces response for download file.  
264 -// it prepares the download response header automatically.  
265 -func (output *BeegoOutput) Download(file string, filename ...string) {  
266 - // check get file error, file not found or other error.  
267 - if _, err := os.Stat(file); err != nil {  
268 - http.ServeFile(output.Context.ResponseWriter, output.Context.Request, file)  
269 - return  
270 - }  
271 -  
272 - var fName string  
273 - if len(filename) > 0 && filename[0] != "" {  
274 - fName = filename[0]  
275 - } else {  
276 - fName = filepath.Base(file)  
277 - }  
278 - output.Header("Content-Disposition", "attachment; filename="+url.PathEscape(fName))  
279 - output.Header("Content-Description", "File Transfer")  
280 - output.Header("Content-Type", "application/octet-stream")  
281 - output.Header("Content-Transfer-Encoding", "binary")  
282 - output.Header("Expires", "0")  
283 - output.Header("Cache-Control", "must-revalidate")  
284 - output.Header("Pragma", "public")  
285 - http.ServeFile(output.Context.ResponseWriter, output.Context.Request, file)  
286 -}  
287 -  
288 -// ContentType sets the content type from ext string.  
289 -// MIME type is given in mime package.  
290 -func (output *BeegoOutput) ContentType(ext string) {  
291 - if !strings.HasPrefix(ext, ".") {  
292 - ext = "." + ext  
293 - }  
294 - ctype := mime.TypeByExtension(ext)  
295 - if ctype != "" {  
296 - output.Header("Content-Type", ctype)  
297 - }  
298 -}  
299 -  
300 -// SetStatus sets response status code.  
301 -// It writes response header directly.  
302 -func (output *BeegoOutput) SetStatus(status int) {  
303 - output.Status = status  
304 -}  
305 -  
306 -// IsCachable returns boolean of this request is cached.  
307 -// HTTP 304 means cached.  
308 -func (output *BeegoOutput) IsCachable() bool {  
309 - return output.Status >= 200 && output.Status < 300 || output.Status == 304  
310 -}  
311 -  
312 -// IsEmpty returns boolean of this request is empty.  
313 -// HTTP 201,204 and 304 means empty.  
314 -func (output *BeegoOutput) IsEmpty() bool {  
315 - return output.Status == 201 || output.Status == 204 || output.Status == 304  
316 -}  
317 -  
318 -// IsOk returns boolean of this request runs well.  
319 -// HTTP 200 means ok.  
320 -func (output *BeegoOutput) IsOk() bool {  
321 - return output.Status == 200  
322 -}  
323 -  
324 -// IsSuccessful returns boolean of this request runs successfully.  
325 -// HTTP 2xx means ok.  
326 -func (output *BeegoOutput) IsSuccessful() bool {  
327 - return output.Status >= 200 && output.Status < 300  
328 -}  
329 -  
330 -// IsRedirect returns boolean of this request is redirection header.  
331 -// HTTP 301,302,307 means redirection.  
332 -func (output *BeegoOutput) IsRedirect() bool {  
333 - return output.Status == 301 || output.Status == 302 || output.Status == 303 || output.Status == 307  
334 -}  
335 -  
336 -// IsForbidden returns boolean of this request is forbidden.  
337 -// HTTP 403 means forbidden.  
338 -func (output *BeegoOutput) IsForbidden() bool {  
339 - return output.Status == 403  
340 -}  
341 -  
342 -// IsNotFound returns boolean of this request is not found.  
343 -// HTTP 404 means not found.  
344 -func (output *BeegoOutput) IsNotFound() bool {  
345 - return output.Status == 404  
346 -}  
347 -  
348 -// IsClientError returns boolean of this request client sends error data.  
349 -// HTTP 4xx means client error.  
350 -func (output *BeegoOutput) IsClientError() bool {  
351 - return output.Status >= 400 && output.Status < 500  
352 -}  
353 -  
354 -// IsServerError returns boolean of this server handler errors.  
355 -// HTTP 5xx means server internal error.  
356 -func (output *BeegoOutput) IsServerError() bool {  
357 - return output.Status >= 500 && output.Status < 600  
358 -}  
359 -  
360 -func stringsToJSON(str string) string {  
361 - var jsons bytes.Buffer  
362 - for _, r := range str {  
363 - rint := int(r)  
364 - if rint < 128 {  
365 - jsons.WriteRune(r)  
366 - } else {  
367 - jsons.WriteString("\\u")  
368 - if rint < 0x100 {  
369 - jsons.WriteString("00")  
370 - } else if rint < 0x1000 {  
371 - jsons.WriteString("0")  
372 - }  
373 - jsons.WriteString(strconv.FormatInt(int64(rint), 16))  
374 - }  
375 - }  
376 - return jsons.String()  
377 -}  
378 -  
379 -// Session sets session item value with given key.  
380 -func (output *BeegoOutput) Session(name interface{}, value interface{}) {  
381 - output.Context.Input.CruSession.Set(name, value)  
382 -}  
1 -package param  
2 -  
3 -import (  
4 - "fmt"  
5 - "reflect"  
6 -  
7 - beecontext "github.com/astaxie/beego/context"  
8 - "github.com/astaxie/beego/logs"  
9 -)  
10 -  
11 -// ConvertParams converts http method params to values that will be passed to the method controller as arguments  
12 -func ConvertParams(methodParams []*MethodParam, methodType reflect.Type, ctx *beecontext.Context) (result []reflect.Value) {  
13 - result = make([]reflect.Value, 0, len(methodParams))  
14 - for i := 0; i < len(methodParams); i++ {  
15 - reflectValue := convertParam(methodParams[i], methodType.In(i), ctx)  
16 - result = append(result, reflectValue)  
17 - }  
18 - return  
19 -}  
20 -  
21 -func convertParam(param *MethodParam, paramType reflect.Type, ctx *beecontext.Context) (result reflect.Value) {  
22 - paramValue := getParamValue(param, ctx)  
23 - if paramValue == "" {  
24 - if param.required {  
25 - ctx.Abort(400, fmt.Sprintf("Missing parameter %s", param.name))  
26 - } else {  
27 - paramValue = param.defaultValue  
28 - }  
29 - }  
30 -  
31 - reflectValue, err := parseValue(param, paramValue, paramType)  
32 - if err != nil {  
33 - logs.Debug(fmt.Sprintf("Error converting param %s to type %s. Value: %v, Error: %s", param.name, paramType, paramValue, err))  
34 - ctx.Abort(400, fmt.Sprintf("Invalid parameter %s. Can not convert %v to type %s", param.name, paramValue, paramType))  
35 - }  
36 -  
37 - return reflectValue  
38 -}  
39 -  
40 -func getParamValue(param *MethodParam, ctx *beecontext.Context) string {  
41 - switch param.in {  
42 - case body:  
43 - return string(ctx.Input.RequestBody)  
44 - case header:  
45 - return ctx.Input.Header(param.name)  
46 - case path:  
47 - return ctx.Input.Query(":" + param.name)  
48 - default:  
49 - return ctx.Input.Query(param.name)  
50 - }  
51 -}  
52 -  
53 -func parseValue(param *MethodParam, paramValue string, paramType reflect.Type) (result reflect.Value, err error) {  
54 - if paramValue == "" {  
55 - return reflect.Zero(paramType), nil  
56 - }  
57 - parser := getParser(param, paramType)  
58 - value, err := parser.parse(paramValue, paramType)  
59 - if err != nil {  
60 - return result, err  
61 - }  
62 -  
63 - return safeConvert(reflect.ValueOf(value), paramType)  
64 -}  
65 -  
66 -func safeConvert(value reflect.Value, t reflect.Type) (result reflect.Value, err error) {  
67 - defer func() {  
68 - if r := recover(); r != nil {  
69 - var ok bool  
70 - err, ok = r.(error)  
71 - if !ok {  
72 - err = fmt.Errorf("%v", r)  
73 - }  
74 - }  
75 - }()  
76 - result = value.Convert(t)  
77 - return  
78 -}  
1 -package param  
2 -  
3 -import (  
4 - "fmt"  
5 - "strings"  
6 -)  
7 -  
8 -//MethodParam keeps param information to be auto passed to controller methods  
9 -type MethodParam struct {  
10 - name string  
11 - in paramType  
12 - required bool  
13 - defaultValue string  
14 -}  
15 -  
16 -type paramType byte  
17 -  
18 -const (  
19 - param paramType = iota  
20 - path  
21 - body  
22 - header  
23 -)  
24 -  
25 -//New creates a new MethodParam with name and specific options  
26 -func New(name string, opts ...MethodParamOption) *MethodParam {  
27 - return newParam(name, nil, opts)  
28 -}  
29 -  
30 -func newParam(name string, parser paramParser, opts []MethodParamOption) (param *MethodParam) {  
31 - param = &MethodParam{name: name}  
32 - for _, option := range opts {  
33 - option(param)  
34 - }  
35 - return  
36 -}  
37 -  
38 -//Make creates an array of MethodParmas or an empty array  
39 -func Make(list ...*MethodParam) []*MethodParam {  
40 - if len(list) > 0 {  
41 - return list  
42 - }  
43 - return nil  
44 -}  
45 -  
46 -func (mp *MethodParam) String() string {  
47 - options := []string{}  
48 - result := "param.New(\"" + mp.name + "\""  
49 - if mp.required {  
50 - options = append(options, "param.IsRequired")  
51 - }  
52 - switch mp.in {  
53 - case path:  
54 - options = append(options, "param.InPath")  
55 - case body:  
56 - options = append(options, "param.InBody")  
57 - case header:  
58 - options = append(options, "param.InHeader")  
59 - }  
60 - if mp.defaultValue != "" {  
61 - options = append(options, fmt.Sprintf(`param.Default("%s")`, mp.defaultValue))  
62 - }  
63 - if len(options) > 0 {  
64 - result += ", "  
65 - }  
66 - result += strings.Join(options, ", ")  
67 - result += ")"  
68 - return result  
69 -}