go+postgresql服务器

  1 func AddGroup(w http.ResponseWriter, req *http.Request) {
  2 
  3     outputReq(req)
  4 
  5     responseJson := map[string]interface{}{jsonKeyMessage: "", jsonKeyResult: nil}
  6 
  7     initResponse(w)
  8     err := req.ParseForm()
  9     if err != nil {
 10         writeResponse(err.Error(), responseJson, w, http.StatusInternalServerError)
 11         return
 12     }
 13 
 14     nickname := toString(req.Form["nickname"])
 15     delete(req.Form, "nickname")
 16     icon := toString(req.Form["icon"])
 17     delete(req.Form, "icon")
 18 
 19     requestJson := initRequestJson(req.Form)
 20     customInfo, err := json.Marshal(requestJson)
 21     if err != nil {
 22         writeResponse(err.Error(), responseJson, w, http.StatusInternalServerError)
 23         return
 24     }
 25 
 26     db, err := openSql()
 27     defer db.Close()
 28     if err != nil {
 29         writeResponse(err.Error(), responseJson, w, http.StatusInternalServerError)
 30         return
 31     }
 32 
 33     tx, err := db.Begin()
 34     if err != nil {
 35         writeResponse(err.Error(), responseJson, w, http.StatusInternalServerError)
 36         return
 37     }
 38 
 39     stmt, err := tx.Prepare(`
 40     insert into objects(nickname,icon,custom,isUser)
 41     values ($1,$2,$3,false) returning id
 42     `)
 43     if err != nil {
 44         writeResponse(err.Error(), responseJson, w, http.StatusInternalServerError)
 45         return
 46     }
 47 
 48     rows, err := stmt.Query(nickname, icon, customInfo)
 49     if err != nil {
 50         errString := err.Error() + " " + errorString(tx.Rollback())
 51         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 52         return
 53     }
 54 
 55     if !rows.Next() {
 56         errString := "add group failed! " + errorString(tx.Rollback())
 57         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 58         return
 59     }
 60 
 61     var groupID int64
 62     err = rows.Scan(&groupID)
 63     ///fmt.Println("groupID:", +groupID)
 64     if err != nil {
 65         errString := err.Error() + " " + errorString(tx.Rollback())
 66         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 67         return
 68     }
 69 
 70     err = rows.Close()
 71     if err != nil {
 72         errString := err.Error() + " " + errorString(tx.Rollback())
 73         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 74         return
 75     }
 76     err = stmt.Close()
 77     if err != nil {
 78         errString := err.Error() + " " + errorString(tx.Rollback())
 79         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 80         return
 81     }
 82     stmt, err = tx.Prepare("insert into groups(id,createTime) values($1,now())")
 83     defer stmt.Close()
 84     if err != nil {
 85         errString := err.Error() + " " + errorString(tx.Rollback())
 86         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 87         return
 88     }
 89 
 90     result, err := stmt.Exec(groupID)
 91     //rows1, err := tx.Exec("insert into groups(id,createTime) values('" + fmt.Sprint(groupID) + "',now())")
 92     if err != nil {
 93         errString := err.Error() + " " + errorString(tx.Rollback())
 94         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
 95         return
 96     }
 97 
 98     affect, err := result.RowsAffected()
 99     if err != nil {
100         errString := err.Error() + " " + errorString(tx.Rollback())
101         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
102         return
103     }
104     if affect != 1 {
105         errString := "add group failed! " + errorString(tx.Rollback())
106         writeResponse(errString, responseJson, w, http.StatusInternalServerError)
107         return
108     }
109 
110     _, err = tx.Exec("commit")
111     if err != nil {
112         writeResponse(err.Error(), responseJson, w, http.StatusInternalServerError)
113         return
114     }
115     responseJson[jsonKeyResult] = fmt.Sprint(groupID)
116     writeResponse("success", responseJson, w, http.StatusOK)
117 }
postgresql事务

 上面代码有一个大bug,事务最后执行tx.Exec("commit"),然后没有执行tx.Commit(),导致事务没有关闭,然后每次一执行就产生一个连接postgresql的tcp,一直没关闭,在线程里看到的是(idle)死进程,最后导致postgresql太多客户端,然后报错FATAL: sorry, too many clients already 

用root进入psql,执行

SELECT COUNT(*) from pg_stat_activity;

发现结果是100(默认最大连接)

执行sudo service postgresql-9.6重启postgresql,然后找代码bug

 解决:使用连接池channal,限定最大连接数,或者只用一个db,每次用的时候都加互斥锁

posted @ 2017-03-26 15:52  cdongyang  阅读(468)  评论(0编辑  收藏  举报