多个response.WriteHeader调用

时间:2016-02-24 13:09:06

标签: go go-templates

我是新手,很难重新制作模板。

以下是我要生成模板的函数:

base.html文件

//Render templates for the given name, template definition and data object
func renderTemplate(w http.ResponseWriter, name string, template string, viewModel interface{}) {
    // Ensure the template exists in the map.
    tmpl, ok := templates[name]
    if !ok {
        http.Error(w, "The template does not exist.", http.StatusInternalServerError)
    }
    err := tmpl.ExecuteTemplate(w, template, viewModel)
    if err != nil {
    log.Printf("temlate error here")
        http.Error(w, err.Error(), http.StatusInternalServerError)
    }
}


 func EditNote(w http.ResponseWriter, r *http.Request) {
    //var viewModel models.EditChnl
    vars := mux.Vars(r)
    //ch := bson.M{"id": "Ale"}
    title := vars["title"]
    log.Printf("%v\n", title)
    session, err := mgo.Dial("localhost")
    if err != nil {
      panic(err)
    }
        defer session.Close()
        session.SetMode(mgo.Monotonic, true)
        c := session.DB("tlgdb").C("chnls")
        log.Printf("title is %s \n", title)
        var result  []models.Chnl
        err = c.Find(bson.M{"title": "xxx"}).All(&result)
        log.Printf("%v\n", result)

        if err != nil {
          log.Printf("doc not found")
          log.Fatal(err)
          return
        }
    renderTemplate(w, "edit", "base", result)
}

以下是模板:

{{define "base"}}
<html>
  <head>
    {{template "head" .}}
  </head>
  <body>

    {{template "body" .}}

  </body>
</html>
{{end}}

edit.thml

{{define "head"}}<title>Edit Note</title>{{end}}
{{define "body"}}
<h1>Edit Note</h1>
<form action="/chnls/update/{{.Title}}" method="post">
<p>Title:<br> <input type="text" value="{{.Title}}" name="title"></p>
<p>Description:<br> <textarea rows="4" cols="50" name="description">{{.Description}}</textarea> </p>
<p><input type="submit" value="submit"/></p>
</form>
{{end}}

要渲染的对象是:

type Chnl struct {
    Id    bson.ObjectId `json:"id"  bson:"_id,omitempty"`
    Title       string
    Description string
    CreatedOn   time.Time
    Creator     string
    Visits  int
    Score       int
}

我正在尝试渲染的对象存在于mongodb中,我可以在控制台中将其打印出来:

[{ObjectIdHex("56cc4493bc54f4245cb4d36b") sometitle blabla 2016-02-23 12:37:55.972 +0100 CET blabla 0 0}]

但是我收到了这个错误:

temlate error here
http: multiple response.WriteHeader calls

我想知道这里有什么问题以及如何解决它?

1 个答案:

答案 0 :(得分:2)

根本问题是您将一片Chnl传递给模板:

var result  []models.Chnl
// ...
renderTemplate(w, "edit", "base", result)

renderTemplate()内的viewModel param值将为result

在模板中,您可以引用点的字段,如果它是Chnl值而不是它的一部分:{{.Title}}。因此,解决它的第一次尝试将失败。

记录错误很有用,因此更改日志记录也会打印实际错误,而不仅仅是一般错误:

log.Printf("Temlate error here: %v", err)

结果是:

  

2016/02/24 14:57:09这里的Temlate错误:template:edit.html:4:30:在&lt; .Title&gt;处执行“body”:无法评估类型[] main中的字段标题。 CHNL

我认为你只想传递1 Chnl值而不是它们的一部分。在EditNote()

if len(result) > 0 {
    renderTemplate(w, "edit", "base", result[0])
}

接下来,知道http.Error()将内容写入响应。这意味着您无法将更多标头值写入响应。通常当您在处理程序中调用http.Error()时,您应该返回而不对响应做任何事情:

if !ok {
    http.Error(w, "The template does not exist.", http.StatusInternalServerError)
    return // NOTE THIS RETURN
}

同样,在所有http.Error()来电后,请插入return。您可以进行一些清理,但不应在http.Error()之后触及响应。