{"id":203328,"date":"2013-11-23T00:29:02","date_gmt":"2013-11-22T20:29:02","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=203328"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=203328","title":{"rendered":"<span class=\"post_title\">\u041d\u0430\u0433\u0440\u0443\u0437\u043e\u0447\u043d\u044b\u0439 \u0442\u0435\u0441\u0442 \u043d\u0430 Go, \u0432\u0435\u0440\u0441\u0438\u044f 2<\/span>"},"content":{"rendered":"<div class=\"content html_format\">   \t\u041d\u0438\u043a\u0430\u043a \u043d\u0435 \u0434\u043e\u0445\u043e\u0434\u0438\u043b\u0438 \u0440\u0443\u043a\u0438 \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u0442\u044c <a href=\"http:\/\/habrahabr.ru\/post\/187212\/\">go-meter<\/a>. \u0423\u0432\u0435\u043b\u0438\u0447\u0438\u0442\u044c \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c, \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u0431\u043e\u043b\u0435\u0435 \u043f\u043e\u043b\u043d\u044b\u0439 \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u043c \u0438 \u0434\u043e\u0432\u0435\u0441\u0442\u0438 \u0434\u043e \u043f\u0440\u0438\u0431\u043b\u0438\u0436\u0435\u043d\u0438\u044f \u043a <a href=\"https:\/\/github.com\/wg\/wrk\">wrk<\/a>. \u0412 \u0438\u0434\u0435\u0430\u043b\u0435 \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0443\u0432\u0438\u0434\u0435\u0442\u044c \u043b\u0435\u0433\u043a\u043e \u0438 \u0443\u0434\u043e\u0431\u043d\u043e \u0440\u0430\u0441\u0448\u0438\u0440\u044f\u0435\u043c\u0443\u044e \u0430\u043b\u044c\u0442\u0435\u0440\u043d\u0430\u0442\u0438\u0432\u0443. \u0414\u0430, \u0432 <a href=\"https:\/\/github.com\/wg\/wrk\">wrk<\/a> \u043d\u0435\u0434\u0430\u0432\u043d\u043e \u043f\u043e\u044f\u0432\u0438\u043b\u0430\u0441\u044c \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0430 <a href=\"https:\/\/github.com\/wg\/wrk\/commit\/c6679dc58a72dd468667d5e43e55aad4b585d621\">Lua \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432<\/a>, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0440\u0435\u0448\u0430\u044e\u0442 \u043c\u043d\u043e\u0433\u0438\u0435 \u043d\u0435\u0443\u0434\u043e\u0431\u0441\u0442\u0432\u0430, \u043d\u043e \u0438 \u0442\u0430\u043c \u0442\u043e\u0436\u0435 \u0435\u0441\u0442\u044c \u043d\u0435\u043f\u0440\u0438\u044f\u0442\u043d\u044b\u0435 \u043d\u044e\u0430\u043d\u0441\u044b, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0440\u0430\u0441\u0448\u0438\u0440\u0435\u043d\u043d\u0443\u044e \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0443 \u0441\u043e\u0431\u0438\u0440\u0430\u0442\u044c \u043d\u0435 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0441\u044f, \u0442\u0430\u043a \u043a\u0430\u043a \u043c\u0435\u0442\u043e\u0434\u044b \u0432\u044b\u0432\u043e\u0434\u0430 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438 \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043d\u0430 \u043f\u0435\u0440\u0432\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435, \u0438 \u043a \u0441\u043e\u0431\u0440\u0430\u043d\u043d\u044b\u043c \u0434\u0430\u043d\u043d\u044b\u043c \u043d\u0430 \u0434\u0440\u0443\u0433\u0438\u0445 \u043f\u043e\u0442\u043e\u043a\u0430\u0445 \u0434\u043e\u0441\u0442\u0443\u043f\u0430 \u043d\u0435\u0442, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u0441\u0432\u043e\u0434\u0438\u0442\u0441\u044f \u043e\u043f\u044f\u0442\u044c \u043a \u0442\u043e\u043c\u0443, \u0447\u0442\u043e-\u0431\u044b \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u0442\u044c\u0441\u044f \u0432 \u0438\u0441\u0445\u043e\u0434\u043d\u0438\u043a\u0430\u0445 \u0438 \u0434\u0435\u043b\u0430\u0442\u044c \u043f\u043e\u0434 \u0441\u0435\u0431\u044f, \u043d\u043e \u044d\u0442\u043e \u043d\u0435 \u0442\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u0430\u044f \u0437\u0430\u0434\u0430\u0447\u0430. \u0418 \u0442\u0430\u043a, \u0433\u043e\u0442\u043e\u0432\u0438\u043c \u043d\u0430\u0433\u0440\u0443\u0437\u043e\u0447\u043d\u044b\u0439 \u0442\u0435\u0441\u0442 \u043d\u0430 Go, c \u043f\u043b\u044e\u0448\u043a\u0430\u043c\u0438. \u041a\u043e\u043c\u0443 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e, \u043f\u0440\u043e\u0448\u0443 \u043f\u043e\u0434 \u043a\u0430\u0442.<br \/>  <a name=\"habracut\"><\/a>  <\/p>\n<h5>\u0427\u0442\u043e \u0435\u0441\u0442\u044c \u0438 \u0447\u0442\u043e \u043d\u0443\u0436\u043d\u043e<\/h5>\n<p>  \u0421 \u043d\u0430\u0447\u0430\u043b\u0430 \u0440\u0430\u0437\u0431\u0435\u0440\u0435\u043c\u0441\u044f \u0447\u0442\u043e \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e:<br \/>   \u2014 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 GET\/POST\/PUT\/DELETE \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432<br \/>   \u2014 \u043f\u0435\u0440\u0435\u0431\u043e\u0440 URL, \u0438 POST body<br \/>   \u2014 \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u043e\u0442\u043a\u0440\u044b\u0442\u044b\u043c\u0438 \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u044f\u043c\u0438<br \/>   \u2014 \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u043f\u043e\u0442\u043e\u043a\u0430\u043c\u0438<br \/>   \u2014 \u0443\u043a\u0430\u0437\u0430\u043d\u0438\u0435 \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438 \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f<br \/>   \u2014 \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u044f \u043f\u043e \u043c\u0430\u043a\u0441\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u043c\u0443 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u0443 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443<br \/>   \u2014 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0438\u0441\u043a\u043b\u044e\u0447\u0438\u0442\u044c \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043f\u0435\u0440\u0432\u044b\u0445 \u0441\u0435\u043a\u0443\u043d\u0434 \u0438\u0437 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438, \u0447\u0442\u043e\u0431\u044b \u0438\u0437\u0431\u0435\u0436\u0430\u0442\u044c \u0438\u0441\u043a\u0430\u0436\u0435\u043d\u0438\u044f \u0432 \u043c\u043e\u043c\u0435\u043d\u0442 \u043f\u0440\u043e\u0433\u0440\u0435\u0432\u0430 HTTP \u0441\u0435\u0440\u0432\u0435\u0440\u0430<\/p>\n<h5>\u041f\u043b\u0430\u043d<\/h5>\n<p>   \u2014 \u043f\u0443\u043b \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0439<br \/>   \u2014 \u043f\u0440\u043e\u0441\u0442\u044b\u0435 Request\/Response<br \/>   \u2014 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0430<br \/>   \u2014 profit  <\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u043c\u044b\u0441\u043b\u0438 \u0432\u0441\u043b\u0443\u0445<\/b><\/p>\n<div class=\"spoiler_text\">\u0420\u0430\u0437 \u043d\u0443\u0436\u043d\u043e \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u044f, \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 http.Client \u043d\u0430\u043c \u043d\u0435 \u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442 (\u0434\u0430 \u0438 \u0431\u043e\u043b\u044c\u0448\u043e\u0439 \u043e\u043d \u0434\u043b\u044f \u0442\u0430\u043a\u043e\u0439 \u0437\u0430\u0434\u0430\u0447\u0438), \u0443\u043c\u0435\u0435\u0442 \u0441\u043b\u0438\u0448\u043a\u043e\u043c \u043c\u043d\u043e\u0433\u043e \u0438\u0437-\u0437\u0430 \u0447\u0435\u0433\u043e \u0441\u0442\u0440\u0430\u0434\u0430\u0435\u0442 \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c. \u0422\u0430\u043a \u043a\u0430\u043a \u0443 \u043d\u0430\u0441 \u043f\u043e\u0434\u0440\u0430\u0437\u0443\u043c\u0435\u0432\u0430\u0435\u0442\u0441\u044f \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u0442\u043e\u043a\u043e\u0432 \u0432\u043e\u0440\u043a\u0435\u0440\u043e\u0432 \u0434\u043b\u044f \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0438 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432, \u0442\u043e \u043d\u0430\u043c \u043d\u0443\u0436\u0435\u043d \u043f\u0443\u043b \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0439, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043e\u043d\u0438 \u0431\u0443\u0434\u0443\u0442 \u043c\u0435\u0436\u0434\u0443 \u0441\u043e\u0431\u043e\u0439 \u0434\u0435\u043b\u0438\u0442\u044c. \u0412\u043e\u0440\u043a\u0435\u0440\u0443 \u0436\u0434\u0430\u0442\u044c \u043e\u0442\u0432\u0435\u0442\u0430 \u043e\u0442 \u0441\u0435\u0440\u0432\u0435\u0440\u0430 \u043d\u0435 \u0438\u043c\u0435\u0435\u0442 \u0441\u043c\u044b\u0441\u043b\u0430, \u043c\u044b \u043f\u0440\u043e\u0441\u0442\u043e \u0442\u0435\u0440\u044f\u0435\u043c \u043d\u0430 \u044d\u0442\u043e \u0434\u0440\u0430\u0433\u043e\u0446\u0435\u043d\u043d\u043e\u0435 \u0432\u0440\u0435\u043c\u044f. \u041a\u0430\u043a \u043e\u0446\u0435\u043d\u0438\u0442\u044c \u043f\u0440\u043e\u0445\u043e\u0434\u044f\u0449\u0438\u0439 \u0442\u0440\u0430\u0444\u0438\u043a? \u0421\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0435 http.Request, http.Respose \u0442\u0430\u043a\u043e\u0439 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u0438 \u043d\u0435 \u0434\u0430\u044e\u0442, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0438\u0445 \u043d\u0435 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0441\u044f, \u0437\u043d\u0430\u0447\u0438\u0442 \u043d\u0443\u0436\u043d\u043e \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u044b\u0432\u0430\u0442\u044c \u043f\u0440\u043e\u0441\u0442\u043e\u0439 Request\/Response, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043d\u0430\u043c \u0432\u0441\u0435 \u043d\u0435\u043e\u0445\u043e\u0434\u0438\u043c\u043e\u0435 \u0434\u0430\u0441\u0442. \u0421\u043e\u0431\u0438\u0440\u0430\u0442\u044c \u0441\u044b\u0440\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438 \u0432 \u043a\u043e\u043d\u0446\u0435 \u0438\u0445 \u0430\u0433\u0440\u0435\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0435 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0441\u044f, \u0442\u0430\u043a \u043a\u0430\u043a \u043f\u0430\u043c\u044f\u0442\u044c \u043d\u0435 \u0440\u0435\u0437\u0438\u043d\u043e\u0432\u0430\u044f. \u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u0442\u0430\u0442\u0443 \u043d\u0430 \u043b\u0435\u0442\u0443.  <\/div>\n<\/div>\n<h4>\u041f\u043e\u0435\u0445\u0430\u043b\u0438<\/h4>\n<p>  \u041f\u0443\u043b \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0439 \u043f\u0438\u0448\u0435\u043c \u043d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u043d\u043e\u0433\u043e \u043a\u0430\u043d\u0430\u043b\u0430. \u0412\u044b\u0433\u043b\u044f\u0434\u0435\u0442\u044c \u043e\u043d \u0431\u0443\u0434\u0435\u0442 \u043a\u0430\u043a \u043f\u0440\u043e\u0441\u0442\u043e\u0439 \u043f\u0443\u043b \u043e\u0431\u044a\u0435\u043a\u0442\u043e\u0432, \u0432\u0437\u044f\u043b\u0438 \u043e\u0431\u044a\u0435\u043a\u0442 \u0438\u0437 \u043a\u0430\u043d\u0430\u043b\u0430, \u043f\u043e\u0440\u0430\u0431\u043e\u0442\u0430\u043b\u0438, \u043f\u043e\u043b\u043e\u0436\u0438\u043b\u0438 \u043e\u0431\u0440\u0430\u0442\u043d\u043e.  <\/p>\n<pre><code class=\"go\">type Connection struct { \tconn    net.Conn \tmanager *ConnectionManager }  type ConnectionManager struct { \tconns  chan *Connection \tconfig *Config }  func NewConnectionManager(config *Config) (result *ConnectionManager) {  \tresult = &ConnectionManager{config: config, conns: make(chan *Connection, config.Connections)} \tfor i := 0; i &lt; config.Connections; i++ { \t\tconnection := &Connection{manager: result} \t\tif connection.Dial() != nil { \t\t\tConnectionErrors++ \t\t} \t\tresult.conns &lt;- connection \t} \treturn }  func (this *ConnectionManager) Get() *Connection { \treturn &lt;-this.conns }  func (this *Connection) Dial() error { \tif this.IsConnected() { \t\tthis.Disconnect() \t} \tconn, err := net.Dial(&quot;tcp4&quot;, this.manager.config.Url.Host) \tif err == nil { \t\tthis.conn = conn \t} \treturn err }  func (this *Connection) Disconnect() { \tthis.conn.Close() \tthis.conn = nil }  func (this *Connection) IsConnected() bool { \treturn this.conn != nil }  func (this *Connection) Return() { \tthis.manager.conns &lt;- this } <\/code><\/pre>\n<p>  Request\/Response \u0442\u0443\u0442 \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u0447\u0438\u0442\u0430\u0442\u044c \u0438\u0441\u0445\u043e\u0434\u043d\u0438\u043a\u0438 Go, \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u043a\u0430\u043a \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043d\u043e \u0442\u0430\u043c, \u0438 \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u0443\u043f\u0440\u043e\u0449\u0435\u043d\u043d\u0443\u044e \u0430\u043d\u0430\u043b\u043e\u0433\u0438\u044e, \u0433\u043b\u0430\u0432\u043d\u044b\u043c \u043e\u0442\u043b\u0438\u0447\u0438\u0435\u043c \u0431\u0443\u0434\u0435\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u043e\u0431\u044a\u0435\u043c \u0442\u0440\u0430\u0444\u0438\u043a\u0430 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0437\u0430\u043f\u0440\u043e\u0441\u0430\/\u043e\u0442\u0432\u0435\u0442\u0430 \u0438 \u0441\u044d\u043a\u043e\u043d\u043e\u043c\u0438\u0442\u044c \u0434\u0440\u0430\u0433\u043e\u0446\u0435\u043d\u043d\u043e\u0435 \u0432\u0440\u0435\u043c\u044f.   <\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">Request<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"go\">type Request struct { \tMethod string  \tURL *url.URL  \tHeader map[string][]string  \tBody          io.Reader \tContentLength int64  \tHost string  \tBufferSize int64 }  func (req *Request) Write(w io.Writer) error {  \tbw := &bytes.Buffer{}  \tfmt.Fprintf(bw, &quot;%s %s HTTP\/1.1\\r\\n&quot;, valueOrDefault(req.Method, &quot;GET&quot;), req.URL.RequestURI()) \tfmt.Fprintf(bw, &quot;Host: %s\\r\\n&quot;, req.Host)  \tuserAgent := &quot;&quot; \tif req.Header != nil { \t\tif ua := req.Header[&quot;User-Agent&quot;]; len(ua) &gt; 0 { \t\t\tuserAgent = ua[0] \t\t} \t} \tif userAgent != &quot;&quot; { \t\tfmt.Fprintf(bw, &quot;User-Agent: %s\\r\\n&quot;, userAgent) \t}  \tif req.Method == &quot;POST&quot; || req.Method == &quot;PUT&quot; { \t\tfmt.Fprintf(bw, &quot;Content-Length: %d\\r\\n&quot;, req.ContentLength) \t}  \tif req.Header != nil { \t\tfor key, values := range req.Header { \t\t\tif key == &quot;User-Agent&quot; || key == &quot;Content-Length&quot; || key == &quot;Host&quot; { \t\t\t\tcontinue \t\t\t} \t\t\tfor _, value := range values { \t\t\t\tfmt.Fprintf(bw, &quot;%s: %s\\r\\n&quot;, key, value) \t\t\t} \t\t} \t}  \tio.WriteString(bw, &quot;\\r\\n&quot;)  \tif req.Method == &quot;POST&quot; || req.Method == &quot;PUT&quot; { \t\tbodyReader := bufio.NewReader(req.Body) \t\t_, err := bodyReader.WriteTo(bw) \t\tif err != nil { \t\t\treturn err \t\t} \t} \treq.BufferSize = int64(bw.Len()) \t_, err := bw.WriteTo(w) \treturn err } <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  <\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">Response<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"go\">type Response struct { \tStatus     string \tStatusCode int  \tHeader map[string][]string  \tContentLength int64  \tBufferSize int64 }  func ReadResponse(r *bufio.Reader) (*Response, error) { \ttp := textproto.NewReader(r) \tresp := &Response{}  \tline, err := tp.ReadLine() \tif err != nil { \t\treturn nil, err \t} \tf := strings.SplitN(line, &quot; &quot;, 3) \tresp.BufferSize += int64(len(f) + 2)  \tif len(f) &lt; 2 { \t\treturn nil, errors.New(&quot;Response Header ERROR&quot;) \t}  \treasonPhrase := &quot;&quot; \tif len(f) &gt; 2 { \t\treasonPhrase = f[2] \t} \tresp.Status = f[1] + &quot; &quot; + reasonPhrase \tresp.StatusCode, err = strconv.Atoi(f[1]) \tif err != nil { \t\treturn nil, errors.New(&quot;malformed HTTP status code&quot;) \t}  \tresp.Header = make(map[string][]string) \tfor { \t\tline, err := tp.ReadLine() \t\tif err != nil { \t\t\treturn nil, errors.New(&quot;Response Header ERROR&quot;) \t\t} \t\tresp.BufferSize += int64(len(line) + 2) \t\tif len(line) == 0 { \t\t\tbreak \t\t} else { \t\t\tf := strings.SplitN(line, &quot;:&quot;, 2) \t\t\tresp.Header[f[0]] = append(resp.Header[strings.TrimSpace(f[0])], strings.TrimSpace(f[1])) \t\t} \t}  \tif cl := resp.Header[&quot;Content-Length&quot;]; len(cl) &gt; 0 { \t\ti, err := strconv.ParseInt(cl[0], 10, 0) \t\tif err == nil { \t\t\tresp.ContentLength = i \t\t} \t}  \tbuff := make([]byte, resp.ContentLength) \tr.Read(buff) \tresp.BufferSize += int64(resp.ContentLength)  \treturn resp, nil } <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0414\u043b\u044f \u0442\u043e\u0433\u043e \u0447\u0442\u043e \u0431\u044b \u043d\u0430\u0448\u0438 \u043f\u043e\u0442\u043e\u043a\u0438 \u0432\u044b\u043a\u043b\u044e\u0447\u0438\u043b\u0438\u0441\u044c, \u043a\u043e\u0433\u0434\u0430 \u0432\u0440\u0435\u043c\u044f \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0437\u0430\u043a\u043e\u043d\u0447\u0438\u0442\u0441\u044f, \u0441\u0434\u0435\u043b\u0430\u0435\u043c \u043a\u0430\u043d\u0430\u043b \u0434\u043b\u044f \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u044f \u0440\u0430\u0431\u043e\u0442\u044b \u043f\u043e\u0442\u043e\u043a\u043e\u0432 \u0438 \u043a\u0430\u043d\u0430\u043b, \u043f\u043e \u043a\u043e\u0442\u043e\u0440\u043e\u043c\u0443 \u043a\u0430\u0436\u0434\u044b\u0439 \u043f\u043e\u0442\u043e\u043a \u0431\u0443\u0434\u0435\u0442 \u0441\u043e\u043e\u0431\u0449\u0430\u0442\u044c, \u0447\u0442\u043e \u043e\u043d \u043a\u043e\u0440\u0440\u0435\u043a\u0442\u043d\u043e \u0437\u0430\u0432\u0435\u0440\u0448\u0438\u043b \u0441\u0432\u043e\u044e \u0440\u0430\u0431\u043e\u0442\u0443  <\/p>\n<pre><code class=\"go\">WorkerQuit := make(chan bool, *_threads) WorkerQuited := make(chan bool, *_threads) <\/code><\/pre>\n<p>  \u0437\u0430\u0441\u0435\u0447\u0435\u043c \u0432\u0440\u0435\u043c\u044f, \u0438 \u0442\u0430\u043a\u0436\u0435 \u0431\u0443\u0434\u0435\u043c \u0436\u0434\u0430\u0442\u044c Ctr+C(SIGTERM), \u0447\u0442\u043e\u0431\u044b \u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u043c\u043e\u0433\u043b\u043e \u0437\u0430\u0432\u0435\u0440\u0448\u0438\u0442\u044c \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0432 \u043b\u044e\u0431\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442  <\/p>\n<pre><code class=\"go\">\/\/Start Ctr+C listen signalChan := make(chan os.Signal, 1) signal.Notify(signalChan, os.Interrupt, syscall.SIGTERM)  \/\/Wait timers or SIGTERM select { case &lt;-time.After(config.Duration): case &lt;-signalChan: } for i := 0; i &lt; config.Threads; i++ { \tconfig.WorkerQuit &lt;- true } \/\/Wait for threads complete for i := 0; i &lt; config.Threads; i++ { \t&lt;-config.WorkerQuited } <\/code><\/pre>\n<p>  \u0422\u0435\u043f\u0435\u0440\u044c \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043d\u0430 \u0441\u0430\u043c \u0432\u043e\u0440\u043a\u0435\u0440: \u0434\u043b\u044f \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u044f \u043f\u043e \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u0443 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443 \u0432\u043e\u0437\u044c\u043c\u0435\u043c \u0434\u043b\u044f \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0435\u0433\u043e \u0434\u043e\u043b\u044e \u043e\u0442 \u043e\u0431\u0449\u0435\u0433\u043e \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u0430, 4 \u0440\u0430\u0437\u0430 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443 \u0431\u0443\u0434\u0435\u043c \u0443\u0432\u0435\u043b\u0438\u0447\u0438\u0432\u0430\u0442\u044c \u0441\u0447\u0435\u0442\u0447\u0438\u043a \u0438 \u0436\u0434\u0430\u0442\u044c \u043b\u0438\u0431\u043e \u043e\u0441\u0432\u043e\u0431\u043e\u0434\u0438\u0432\u0448\u0438\u0435\u0441\u044f \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \u043b\u0438\u0431\u043e \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u0435 \u0440\u0430\u0431\u043e\u0442\u044b  <\/p>\n<pre><code class=\"go\">func NewThread(config *Config) { \ttimerAllow := time.NewTicker(time.Duration(250) * time.Millisecond) \tallow := int32(config.MRQ \/ 4 \/ config.Threads) \tif config.MRQ == -1 { \t\tallow = 2147483647 \t} else if allow &lt;= 0 { \t\tallow = 1 \t} \tvar connectionErrors int32 = 0 \tcurrentAllow := allow \tfor { \t\tselect { \t\t\/\/\u041f\u043e \u0442\u0430\u0439\u043c\u0435\u0440\u0443 \u0432\u044b\u0441\u0442\u0430\u0432\u043b\u044f\u0435\u043c \u0441\u0447\u0435\u0442\u0447\u0438\u043a \u043d\u0430 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u043d\u044b\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \t\tcase &lt;-timerAllow.C: \t\t\tcurrentAllow = allow \t\t\/\/\u041f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0441\u0432\u043e\u0431\u043e\u0434\u043d\u043e\u0435 \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \t\tcase connection := &lt;-config.ConnectionManager.conns: \t\t\tcurrentAllow-- \t\t\t\/\/\u0415\u0441\u043b\u0438 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u043d\u044b\u0435 \u0437\u0430\u043f\u0440\u043e\u0441\u044b \u043a\u043e\u043d\u0447\u0438\u043b\u0438\u0441\u044c - \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \u0432 \u043f\u0443\u043b \t\t\tif currentAllow &lt; 0 { \t\t\t\tconnection.Return() \t\t\t} else { \t\t\t\t\/\/\u0424\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u043c \u0437\u0430\u043f\u0440\u043e\u0441 \t\t\t\treq := getRequest(config.Method, config.Url, config.Source.GetNext()) \t\t\t\t\/\/\u0415\u0441\u043b\u0438 \u043d\u0443\u0436\u043d\u043e \u043f\u0435\u0440\u0435\u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0430\u0442\u044c\u0441\u044f \u043d\u0430 \u043a\u0430\u0436\u0434\u043e\u043c \u0437\u0430\u043f\u0440\u043e\u0441\u0435 \t\t\t\tif config.Reconnect && connection.IsConnected() { \t\t\t\t\tconnection.Disconnect() \t\t\t\t} \t\t\t\t\/\/\u0415\u0441\u043b\u0438 \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \u0440\u0430\u0437\u043e\u0440\u0432\u0430\u043d\u043e, \u0442\u043e \u043f\u0440\u043e\u0431\u0443\u0435\u043c \u0435\u0433\u043e \u0432\u043e\u0441\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u044c \t\t\t\tif !connection.IsConnected() { \t\t\t\t\tif connection.Dial() != nil { \t\t\t\t\t\tconnectionErrors++ \t\t\t\t\t} \t\t\t\t} \t\t\t\t\/\/\u041e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0437\u0430\u043f\u0440\u043e\u0441 \u0435\u0441\u043b\u0438 \u0435\u0441\u0442\u044c \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435, \u0438\u043d\u0430\u0447\u0435 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \t\t\t\tif connection.IsConnected() { \t\t\t\t\tgo writeSocket(connection, req, config.RequestStats) \t\t\t\t} else { \t\t\t\t\tconnection.Return() \t\t\t\t} \t\t\t} \t\t\/\/\u0416\u0434\u0435\u043c \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u044f \t\tcase &lt;-config.WorkerQuit: \t\t\t\/\/\u0417\u0430\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c \u043e\u0448\u0438\u0431\u043a\u0438 \u043f\u043e \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u044f\u043c \t\t\tatomic.AddInt32(&ConnectionErrors, connectionErrors) \t\t\t\/\/\u041f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0430\u0435\u043c \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u0435 \t\t\tconfig.WorkerQuited &lt;- true \t\t\treturn \t\t} \t} } <\/code><\/pre>\n<p>  \u041a\u0430\u043a \u0442\u043e\u043b\u044c\u043a\u043e \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \u043e\u0441\u0432\u043e\u0431\u043e\u0434\u0438\u0442\u0441\u044f, \u0444\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u043c \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u0439 \u0437\u0430\u043f\u0440\u043e\u0441 \u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0443 \u0435\u0433\u043e, \u0442\u0430\u043a \u043f\u043e \u043a\u0440\u0443\u0433\u0443 \u043f\u043e\u043a\u0430 \u043d\u0435 \u043a\u043e\u043d\u0447\u0438\u0442\u0441\u044f \u0432\u0440\u0435\u043c\u044f. \u041f\u043e\u0441\u043b\u0435 \u0442\u043e\u0433\u043e \u043a\u0430\u043a \u0437\u0430\u043f\u0440\u043e\u0441 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d, \u0430 \u043e\u0442\u0432\u0435\u0442 \u043f\u0440\u043e\u0447\u0438\u0442\u0430\u043d, \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c \u0432 \u043f\u0443\u043b, \u0438 \u043f\u043e\u0442\u043e\u043a \u0441\u043d\u043e\u0432\u0430 \u043f\u043e\u0434\u0445\u0432\u0430\u0442\u0438\u0442 \u0435\u0433\u043e.  <\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u041e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0437\u0430\u043f\u0440\u043e\u0441\u0430<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"go\">func writeSocket(connection *Connection, req *http.Request, read chan *RequestStats) { \tresult := &RequestStats{} \t\/\/\u041f\u043e \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u044e \u043e\u0431\u044f\u0437\u0430\u0442\u0435\u043b\u044c\u043d\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0441\u0442\u0430\u0442\u0443\u0441 \u0438 \u043e\u0442\u0434\u0430\u0435\u043c \u0441\u043e\u0435\u0434\u0438\u043d\u0435\u043d\u0438\u0435 \u0432 \u043f\u0443\u043b \tdefer func() { \t\tconnection.Return() \t\tread &lt;- result \t}()  \tnow := time.Now() \tconn := connection.conn \tbw := bufio.NewWriter(conn) \t\/\/\u041f\u0438\u0448\u0435\u043c \u0437\u0430\u043f\u0440\u043e\u0441 \terr := req.Write(bw) \tif err != nil { \t\tresult.WriteError = err \t\treturn \t} \terr = bw.Flush() \tif err != nil { \t\tresult.WriteError = err \t\treturn \t} \t\/\/\u0416\u0434\u0435\u043c \u043e\u0442\u0432\u0435\u0442\u0430 \tres, err := http.ReadResponse(bufio.NewReader(conn)) \tif err != nil { \t\tresult.ReadError = err \t\treturn \t} \t\/\/\u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u043d\u0443\u0436\u043d\u0443\u044e \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e \tresult.Duration = time.Now().Sub(now) \tresult.NetOut = req.BufferSize \tresult.NetIn = res.BufferSize \tresult.ResponseCode = res.StatusCode \treq.Body = nil } <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u041e\u0441\u0442\u0430\u043b\u043e\u0441\u044c \u0434\u0435\u043b\u043e \u0437\u0430 \u043c\u0430\u043b\u044b\u043c, \u0441\u043e\u0431\u0440\u0430\u0442\u044c \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0443 \u0438\u0437 \u043e\u0431\u044a\u0435\u043a\u0442\u043e\u0432 RequestStats \u0438 \u043e\u0444\u043e\u0440\u043c\u0438\u0442\u044c \u0435\u0435  <\/p>\n<pre><code class=\"go\">\/\/\u0412\u0441\u044f \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0430 type StatsSource struct { \tReaded          int64 \tWrited          int64 \tRequests        int \tSkiped          int \tMin             time.Duration \tMax             time.Duration \tSum             int64 \tCodes           map[int]int \tDurationPercent map[time.Duration]int \tReadErrors      int \tWriteErrors     int \tWork            time.Duration }  \/\/\u0421\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0430 \u0434\u043b\u044f \u043f\u043e\u0441\u0435\u043a\u0443\u043d\u0434\u043d\u044b\u0445 \u043e\u0442\u0447\u0435\u0442\u043e\u0432 type StatsSourcePerSecond struct { \tReaded   int64 \tWrited   int64 \tRequests int \tSkiped   int \tSum      int64 }  \/\/\u0410\u0433\u0440\u0435\u0433\u0430\u0442\u043e\u0440 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438 func StartStatsAggregator(config *Config) { \tallowStore := true \tallowStoreTime := time.After(config.ExcludeSeconds) \tif config.ExcludeSeconds.Seconds() &gt; 0 { \t\tallowStore = false \t}                 \tverboseTimer := time.NewTicker(time.Duration(1) * time.Second) \tif config.Verbose { \t\tfmt.Printf(&quot;%s %s %s %s %s %s\\n&quot;, \t\t\tnewSpancesFormatRightf(&quot;Second&quot;, 10, &quot;%s&quot;), \t\t\tnewSpancesFormatRightf(&quot;Total&quot;, 10, &quot;%s&quot;), \t\t\tnewSpancesFormatRightf(&quot;Req\/sec&quot;, 10, &quot;%s&quot;), \t\t\tnewSpancesFormatRightf(&quot;Avg\/sec&quot;, 10, &quot;%s&quot;), \t\t\tnewSpancesFormatRightf(&quot;In\/sec&quot;, 10, &quot;%s&quot;), \t\t\tnewSpancesFormatRightf(&quot;Out\/sec&quot;, 10, &quot;%s&quot;), \t\t) \t} else { \t\tverboseTimer.Stop() \t}  \tsource = StatsSource{ \t\tCodes:           make(map[int]int), \t\tDurationPercent: make(map[time.Duration]int), \t}  \tperSecond := StatsSourcePerSecond{}  \tstart := time.Now() \tfor { \t\tselect { \t\t \/\/\u0422\u0430\u0439\u043c\u0435\u0440 \u0434\u043b\u044f \u043f\u043e\u0441\u0435\u043a\u0443\u043d\u0434\u043d\u044b\u0445 \u043e\u0442\u0447\u0435\u0442\u043e\u0432 \t\tcase &lt;-verboseTimer.C: \t\t\tif perSecond.Requests-perSecond.Skiped &gt; 0 && config.Verbose { \t\t\t\t\/\/\u0421\u0447\u0438\u0442\u0430\u0435\u043c \u0441\u0440\u0435\u0434\u043d\u0435\u0435 \u0432\u0440\u0435\u043c\u044f \u043e\u0442\u0432\u0435\u0442\u0430 \t\t\t\tavgMilliseconds := perSecond.Sum \/ int64(perSecond.Requests-perSecond.Skiped) \t\t\t\tavg := time.Duration(avgMilliseconds) * time.Millisecond \t\t\t\t\/\/\u041f\u0438\u0448\u0435\u043c \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0443 \t\t\t\tfmt.Printf(&quot;%s %s %s %s %s %s\\n&quot;, \t\t\t\t\tnewSpancesFormatRightf(roundToSecondDuration(time.Now().Sub(start)), 10, &quot;%v&quot;), \t\t\t\t\tnewSpancesFormatRightf(source.Requests, 10, &quot;%d&quot;), \t\t\t\t\tnewSpancesFormatRightf(perSecond.Requests, 10, &quot;%d&quot;), \t\t\t\t\tnewSpancesFormatRightf(avg, 10, &quot;%v&quot;), \t\t\t\t\tnewSpancesFormatRightf(Bites(perSecond.Readed), 10, &quot;%s&quot;), \t\t\t\t\tnewSpancesFormatRightf(Bites(perSecond.Writed), 10, &quot;%s&quot;), \t\t\t\t) \t\t\t} \t\t\t\/\/\u0421\u0431\u0440\u0430\u0441\u044b\u0432\u0430\u0435\u043c \u0434\u0430\u043d\u043d\u044b\u0435 \t\t\tperSecond = StatsSourcePerSecond{} \t\t\/\/\u0422\u0430\u0439\u043c\u0435\u0440 \u0434\u043b\u044f \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u044f \u0441\u0431\u043e\u0440\u0430 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438 \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u043f\u0440\u043e\u043f\u0443\u0441\u043a\u0430 \u043d\u0430 \u0441\u0442\u0430\u0440\u0442\u0435 \t\tcase &lt;-allowStoreTime: \t\t\tallowStore = true \t\t\/\/\u041f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u043e\u0442\u0432\u0435\u0442 \u043e\u0442 \u0441\u0435\u0440\u0432\u0435\u0440\u0430 \t\tcase res := &lt;-config.RequestStats: \t\t\t\/\/\u0415\u0441\u043b\u0438 \u0431\u044b\u043b\u0438 \u043e\u0448\u0438\u0431\u043a\u0438 - \u043f\u0440\u043e\u0441\u0442\u043e \u0438\u0445 \u0437\u0430\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c, \u043e\u0441\u0442\u0430\u043b\u044c\u043d\u0430\u044f \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044f \u043d\u0430\u043c \u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u0430 \t\t\tif res.ReadError != nil { \t\t\t\tsource.ReadErrors++ \t\t\t\tcontinue \t\t\t} else if res.WriteError != nil { \t\t\t\tsource.WriteErrors++ \t\t\t\tcontinue \t\t\t} \t\t\t\/\/\u0418\u043d\u043a\u0440\u0435\u043c\u0435\u043d\u0442\u0438\u0440\u0443\u0435\u043c \u0441\u0447\u0435\u0442\u0447\u0438\u043a\u0438 \t\t\tsource.Requests++ \t\t\tperSecond.Requests++ \t\t\tperSecond.Readed += res.NetIn \t\t\tperSecond.Writed += res.NetOut \t\t\tsource.Readed += res.NetIn \t\t\tsource.Writed += res.NetOut \t\t\t\/\/\u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0443 \u043f\u043e \u0437\u0430\u043f\u0440\u043e\u0441\u0430\u043c \u0432 \u0440\u0430\u0437\u0440\u0435\u0437\u0435 HTTP \u043a\u043e\u0434\u043e\u0432 \t\t\tsource.Codes[res.ResponseCode]++ \t\t\tif !allowStore { \t\t\t\tperSecond.Skiped++ \t\t\t\tsource.Skiped++ \t\t\t\tcontinue \t\t\t} \t\t\t\/\/\u0414\u043b\u044f \u0441\u0440\u0435\u0434\u043d\u0435\u0433\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043e\u0442\u0432\u0435\u0442\u0430 \t\t\tsum := int64(res.Duration.Seconds() * 1000) \t\t\tsource.Sum += sum \t\t\tperSecond.Sum += sum  \t\t\t\/\/\u041c\u0430\u043a\u0441\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0435 \u0438 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0435 \u0432\u0440\u0435\u043c\u044f \u043e\u0442\u0432\u0435\u0442\u0430 \t\t\tif source.Min &gt; res.Duration { \t\t\t\tsource.Min = roundDuration(res.Duration) \t\t\t} \t\t\tif source.Max &lt; res.Duration { \t\t\t\tsource.Max = roundDuration(res.Duration) \t\t\t} \t\t\t\/\/\u041a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u0432 \u0440\u0430\u0437\u0440\u0435\u0437\u0435 \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043e\u0442\u0432\u0435\u0442\u0430 \u043e\u043a\u0440\u0443\u0433\u043b\u0435\u043d\u043d\u0430\u044f \u0434\u043e 10 \u043c\u0438\u043b\u043b\u0438\u0441\u0435\u043a\u0443\u043d\u0434 \t\t\tduration := time.Duration(res.Duration.Nanoseconds()\/10000000) * time.Millisecond * 10 \t\t\tsource.DurationPercent[duration]++ \t\t\/\/\u0417\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u0435 \u0441\u0431\u043e\u0440\u0430 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438 \t\tcase &lt;-config.StatsQuit: \t\t\t\/\/\u0417\u0430\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c \u043e\u0431\u0449\u0435\u0435 \u0432\u0440\u0435\u043c\u044f \u0442\u0435\u0441\u0442\u0430 \t\t\tsource.Work = time.Duration(time.Now().Sub(start).Seconds()*1000) * time.Millisecond \t\t\tif config.Verbose { \t\t\t\ts := &quot;&quot; \t\t\t\tfor { \t\t\t\t\tif len(s) &gt;= 61 { \t\t\t\t\t\tbreak \t\t\t\t\t} \t\t\t\t\ts += &quot;-&quot; \t\t\t\t} \t\t\t\tfmt.Println(s) \t\t\t} \t\t\t\/\/\u041f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0430\u0435\u043c \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u0435 \t\t\tconfig.StatsQuit &lt;- true \t\t\treturn \t\t} \t} } <\/code><\/pre>\n<h5>\u041f\u043e\u0434\u0432\u043e\u0434\u0438\u043c \u0438\u0442\u043e\u0433\u0438<\/h5>\n<p>  \u041a\u0430\u043a \u043f\u0430\u0440\u0441\u0438\u0442\u044c \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u044b \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u0438 \u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0432\u044b\u0432\u043e\u0434 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438 \u044f \u043e\u043f\u0443\u0449\u0443, \u0442\u0430\u043a \u043a\u0430\u043a \u044d\u0442\u043e \u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e. \u0410 \u0442\u0435\u043f\u0435\u0440\u044c \u0434\u0430\u0432\u0430\u0439\u0442\u0435 \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u043c, \u0447\u0442\u043e \u0443 \u043d\u0430\u0441 \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c. \u0414\u043b\u044f \u043f\u0440\u043e\u0431\u044b \u043d\u0430\u0442\u0440\u0430\u0432\u0438\u043c wrk \u043d\u0430 Node.js \u043a\u043b\u0430\u0441\u0442\u0435\u0440  <\/p>\n<pre><code class=\"bash\">% .\/wrk -c 21 -t 7 -d 30s -L http:\/\/localhost:3001\/index.html Running 30s test @ http:\/\/localhost:3001\/index.html   7 threads and 21 connections   Thread Stats   Avg      Stdev     Max   +\/- Stdev     Latency     1.09ms    6.55ms 152.07ms   99.63%     Req\/Sec     5.20k     3.08k   14.33k    58.75%   Latency Distribution      50%  490.00us      75%    0.89ms      90%    1.83ms      99%    5.04ms   1031636 requests in 30.00s, 153.48MB read Requests\/sec:  34388.25 Transfer\/sec:      5.12MB <\/code><\/pre>\n<p>  \u0438 \u0442\u043e\u0436\u0435 \u0441\u0430\u043c\u043e\u0435 \u043d\u0430 go \u0441 GOMAXPROCS=1  <\/p>\n<pre><code class=\"bash\">% .\/go-meter -t 7 -c 21 -d 30s -u http:\/\/localhost:3001\/index.html     Running test threads: 7, connections: 21 in 30s GET http:\/\/localhost:3001\/index.html Stats:            Min       Avg       Max   Latency           0         0      83ms   843183 requests in 30s, net: in 103MB, out 62MB HTTP Codes:       200       100.00% Latency:                 0        99.99%      10ms - 80ms         0.01% Requests: 28106.10\/sec Net In: 27MBit\/sec Net Out: 17MBit\/sec Transfer: 5.5MB\/sec <\/code><\/pre>\n<p>  \u041f\u043e\u043b\u0443\u0447\u0430\u0435\u043c 28106 \u043f\u0440\u043e\u0442\u0438\u0432 34388 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443 \u2014 \u044d\u0442\u043e \u043f\u0440\u0438\u043c\u0435\u0440\u043d\u043e \u043d\u0430 20% \u043c\u0435\u043d\u044c\u0448\u0435, \u043f\u043e \u0441\u0440\u0430\u0432\u043d\u0435\u043d\u0438\u044e \u0441 \u0447\u0438\u0441\u0442\u044b\u043c C\u0438 + event loop + nio. \u0414\u043e\u0432\u043e\u043b\u044c\u043d\u043e \u043d\u0435\u043f\u043b\u043e\u0445\u043e, \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 GOMAXPROCS \u0440\u0430\u0437\u043d\u0438\u0446\u044b \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043d\u0435\u0442, \u0442\u0430\u043a \u043a\u0430\u043a \u0431\u043e\u043b\u044c\u0448\u0443\u044e \u0447\u0430\u0441\u0442\u044c \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u0440\u043d\u043e\u0433\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043e\u0442\u0431\u0438\u0440\u0430\u0435\u0442 Node.js.<br \/>  \u041c\u0438\u043d\u0443\u0441\u044b:<br \/>   \u2014 \u043f\u043e\u0442\u0435\u0440\u044f 20% \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438, \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043f\u0440\u043e\u0431\u043e\u0432\u0430\u0442\u044c \u0443\u043f\u0440\u043e\u0441\u0442\u0438\u0442\u044c Request\/Response, \u043c\u043e\u0436\u0435\u0442 \u0434\u0430\u0442\u044c \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438<br \/>   \u2014 \u0435\u0449\u0435 \u043d\u0435\u0442 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438 HTTPS<br \/>   \u2014 \u0435\u0449\u0435 \u043d\u0435\u043b\u044c\u0437\u044f \u0443\u043a\u0430\u0437\u0430\u0442\u044c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c\u0441\u043a\u0438\u0435 HTTP \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0438 \u0438 timeout<\/p>\n<p>  \u0412\u0441\u0435 \u0438\u0441\u0445\u043e\u0434\u043d\u0438\u043a\u0438 \u0442\u0443\u0442 \u2014 <a href=\"https:\/\/github.com\/a696385\/go-meter\">Github<\/a><\/p>\n<p>  \u041a\u0430\u043a \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f  <\/p>\n<pre><code class=\"bash\">% go get github.com\/a696385\/go-meter  % $GOPATH\/bin\/go-meter -h  <\/code><\/pre>\n<p>  \u0421\u043f\u0430\u0441\u0438\u0431\u043e \u0437\u0430 \u0432\u043d\u0438\u043c\u0430\u043d\u0438\u0435!    \t<\/p>\n<div class=\"clear\"><\/div>\n<\/p><\/div>\n<p> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"http:\/\/habrahabr.ru\/post\/203328\/\"> http:\/\/habrahabr.ru\/post\/203328\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div class=\"content html_format\">   \t\u041d\u0438\u043a\u0430\u043a \u043d\u0435 \u0434\u043e\u0445\u043e\u0434\u0438\u043b\u0438 \u0440\u0443\u043a\u0438 \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u0442\u044c <a href=\"http:\/\/habrahabr.ru\/post\/187212\/\">go-meter<\/a>. \u0423\u0432\u0435\u043b\u0438\u0447\u0438\u0442\u044c \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c, \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u0431\u043e\u043b\u0435\u0435 \u043f\u043e\u043b\u043d\u044b\u0439 \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u043c \u0438 \u0434\u043e\u0432\u0435\u0441\u0442\u0438 \u0434\u043e \u043f\u0440\u0438\u0431\u043b\u0438\u0436\u0435\u043d\u0438\u044f \u043a <a href=\"https:\/\/github.com\/wg\/wrk\">wrk<\/a>. \u0412 \u0438\u0434\u0435\u0430\u043b\u0435 \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0443\u0432\u0438\u0434\u0435\u0442\u044c \u043b\u0435\u0433\u043a\u043e \u0438 \u0443\u0434\u043e\u0431\u043d\u043e \u0440\u0430\u0441\u0448\u0438\u0440\u044f\u0435\u043c\u0443\u044e \u0430\u043b\u044c\u0442\u0435\u0440\u043d\u0430\u0442\u0438\u0432\u0443. \u0414\u0430, \u0432 <a href=\"https:\/\/github.com\/wg\/wrk\">wrk<\/a> \u043d\u0435\u0434\u0430\u0432\u043d\u043e \u043f\u043e\u044f\u0432\u0438\u043b\u0430\u0441\u044c \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0430 <a href=\"https:\/\/github.com\/wg\/wrk\/commit\/c6679dc58a72dd468667d5e43e55aad4b585d621\">Lua \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432<\/a>, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0440\u0435\u0448\u0430\u044e\u0442 \u043c\u043d\u043e\u0433\u0438\u0435 \u043d\u0435\u0443\u0434\u043e\u0431\u0441\u0442\u0432\u0430, \u043d\u043e \u0438 \u0442\u0430\u043c \u0442\u043e\u0436\u0435 \u0435\u0441\u0442\u044c \u043d\u0435\u043f\u0440\u0438\u044f\u0442\u043d\u044b\u0435 \u043d\u044e\u0430\u043d\u0441\u044b, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0440\u0430\u0441\u0448\u0438\u0440\u0435\u043d\u043d\u0443\u044e \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0443 \u0441\u043e\u0431\u0438\u0440\u0430\u0442\u044c \u043d\u0435 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0441\u044f, \u0442\u0430\u043a \u043a\u0430\u043a \u043c\u0435\u0442\u043e\u0434\u044b \u0432\u044b\u0432\u043e\u0434\u0430 \u0441\u0442\u0430\u0442\u0438\u0441\u0442\u0438\u043a\u0438 \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043d\u0430 \u043f\u0435\u0440\u0432\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435, \u0438 \u043a \u0441\u043e\u0431\u0440\u0430\u043d\u043d\u044b\u043c \u0434\u0430\u043d\u043d\u044b\u043c \u043d\u0430 \u0434\u0440\u0443\u0433\u0438\u0445 \u043f\u043e\u0442\u043e\u043a\u0430\u0445 \u0434\u043e\u0441\u0442\u0443\u043f\u0430 \u043d\u0435\u0442, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u0441\u0432\u043e\u0434\u0438\u0442\u0441\u044f \u043e\u043f\u044f\u0442\u044c \u043a \u0442\u043e\u043c\u0443, \u0447\u0442\u043e-\u0431\u044b \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u0442\u044c\u0441\u044f \u0432 \u0438\u0441\u0445\u043e\u0434\u043d\u0438\u043a\u0430\u0445 \u0438 \u0434\u0435\u043b\u0430\u0442\u044c \u043f\u043e\u0434 \u0441\u0435\u0431\u044f, \u043d\u043e \u044d\u0442\u043e \u043d\u0435 \u0442\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u0430\u044f \u0437\u0430\u0434\u0430\u0447\u0430. \u0418 \u0442\u0430\u043a, \u0433\u043e\u0442\u043e\u0432\u0438\u043c \u043d\u0430\u0433\u0440\u0443\u0437\u043e\u0447\u043d\u044b\u0439 \u0442\u0435\u0441\u0442 \u043d\u0430 Go, c \u043f\u043b\u044e\u0448\u043a\u0430\u043c\u0438. \u041a\u043e\u043c\u0443 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e, \u043f\u0440\u043e\u0448\u0443 \u043f\u043e\u0434 \u043a\u0430\u0442.  <\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-203328","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/203328","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=203328"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/203328\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=203328"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=203328"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=203328"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}