Просмотр исходного кода

Merge pull request #44 from 9seconds/goroutines-number

Decrease the number of used goroutines
tags/0.15^2
Sergey Arkhipov 7 лет назад
Родитель
Сommit
c9f6b922c3
Аккаунт пользователя с таким Email не найден
5 измененных файлов: 35 добавлений и 69 удалений
  1. 3
    4
      Dockerfile
  2. 2
    3
      go.mod
  3. 4
    9
      go.sum
  4. 6
    16
      proxy/proxy.go
  5. 20
    37
      wrappers/conn.go

+ 3
- 4
Dockerfile Просмотреть файл

10
     curl \
10
     curl \
11
     git \
11
     git \
12
     make \
12
     make \
13
-    upx \
14
-  && update-ca-certificates
13
+    upx
15
 
14
 
16
 COPY . /go/src/github.com/9seconds/mtg/
15
 COPY . /go/src/github.com/9seconds/mtg/
17
 
16
 
26
 
25
 
27
 FROM scratch
26
 FROM scratch
28
 
27
 
29
-ENTRYPOINT ["/usr/local/bin/mtg"]
28
+ENTRYPOINT ["/mtg"]
30
 ENV MTG_IP=0.0.0.0 \
29
 ENV MTG_IP=0.0.0.0 \
31
     MTG_PORT=3128 \
30
     MTG_PORT=3128 \
32
     MTG_STATS_IP=0.0.0.0 \
31
     MTG_STATS_IP=0.0.0.0 \
34
 EXPOSE 3128 3129
33
 EXPOSE 3128 3129
35
 
34
 
36
 COPY --from=0 /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/ca-certificates.crt
35
 COPY --from=0 /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/ca-certificates.crt
37
-COPY --from=0 /go/src/github.com/9seconds/mtg/mtg /usr/local/bin/mtg
36
+COPY --from=0 /go/src/github.com/9seconds/mtg/mtg /mtg

+ 2
- 3
go.mod Просмотреть файл

7
 	github.com/davecgh/go-spew v1.1.1 // indirect
7
 	github.com/davecgh/go-spew v1.1.1 // indirect
8
 	github.com/dustin/go-humanize v1.0.0
8
 	github.com/dustin/go-humanize v1.0.0
9
 	github.com/gofrs/uuid v3.1.0+incompatible
9
 	github.com/gofrs/uuid v3.1.0+incompatible
10
-	github.com/juju/errors v0.0.0-20180806074554-22422dad46e1
10
+	github.com/juju/errors v0.0.0-20181012004132-a4583d0a56ea
11
 	github.com/juju/loggo v0.0.0-20180524022052-584905176618 // indirect
11
 	github.com/juju/loggo v0.0.0-20180524022052-584905176618 // indirect
12
 	github.com/juju/testing v0.0.0-20180920084828-472a3e8b2073 // indirect
12
 	github.com/juju/testing v0.0.0-20180920084828-472a3e8b2073 // indirect
13
 	github.com/kr/pretty v0.1.0 // indirect
13
 	github.com/kr/pretty v0.1.0 // indirect
14
-	github.com/kr/pty v1.1.3 // indirect
15
 	github.com/pkg/errors v0.8.0 // indirect
14
 	github.com/pkg/errors v0.8.0 // indirect
16
 	github.com/pmezard/go-difflib v1.0.0 // indirect
15
 	github.com/pmezard/go-difflib v1.0.0 // indirect
17
 	github.com/stretchr/testify v1.2.2
16
 	github.com/stretchr/testify v1.2.2
18
 	go.uber.org/atomic v1.3.2 // indirect
17
 	go.uber.org/atomic v1.3.2 // indirect
19
 	go.uber.org/multierr v1.1.0 // indirect
18
 	go.uber.org/multierr v1.1.0 // indirect
20
 	go.uber.org/zap v1.9.1
19
 	go.uber.org/zap v1.9.1
21
-	golang.org/x/net v0.0.0-20181005035420-146acd28ed58 // indirect
20
+	golang.org/x/net v0.0.0-20181011144130-49bb7cea24b1 // indirect
22
 	gopkg.in/alecthomas/kingpin.v2 v2.2.6
21
 	gopkg.in/alecthomas/kingpin.v2 v2.2.6
23
 	gopkg.in/alexcesaro/statsd.v2 v2.0.0
22
 	gopkg.in/alexcesaro/statsd.v2 v2.0.0
24
 	gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect
23
 	gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect

+ 4
- 9
go.sum Просмотреть файл

6
 github.com/beevik/ntp v0.2.0/go.mod h1:hIHWr+l3+/clUnF44zdK+CWW7fO8dR5cIylAQ76NRpg=
6
 github.com/beevik/ntp v0.2.0/go.mod h1:hIHWr+l3+/clUnF44zdK+CWW7fO8dR5cIylAQ76NRpg=
7
 github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
7
 github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
8
 github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
8
 github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
9
-github.com/dustin/go-humanize v0.0.0-20180713052910-9f541cc9db5d h1:lDrio3iIdNb0Gw9CgH7cQF+iuB5mOOjdJ9ERNJCBgb4=
10
-github.com/dustin/go-humanize v0.0.0-20180713052910-9f541cc9db5d/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
11
 github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo=
9
 github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo=
12
 github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
10
 github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
13
 github.com/gofrs/uuid v3.1.0+incompatible h1:q2rtkjaKT4YEr6E1kamy0Ha4RtepWlQBedyHx0uzKwA=
11
 github.com/gofrs/uuid v3.1.0+incompatible h1:q2rtkjaKT4YEr6E1kamy0Ha4RtepWlQBedyHx0uzKwA=
14
 github.com/gofrs/uuid v3.1.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM=
12
 github.com/gofrs/uuid v3.1.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM=
15
-github.com/juju/errors v0.0.0-20180806074554-22422dad46e1 h1:wnhMXidtb70kDZCeLt/EfsVtkXS5c8zLnE9y/6DIRAU=
16
-github.com/juju/errors v0.0.0-20180806074554-22422dad46e1/go.mod h1:W54LbzXuIE0boCoNJfwqpmkKJ1O4TCTZMetAt6jGk7Q=
13
+github.com/juju/errors v0.0.0-20181012004132-a4583d0a56ea h1:g2k+8WR7cHch4g0tBDhfiEvAp7fXxTNBiD1oC1Oxj3E=
14
+github.com/juju/errors v0.0.0-20181012004132-a4583d0a56ea/go.mod h1:W54LbzXuIE0boCoNJfwqpmkKJ1O4TCTZMetAt6jGk7Q=
17
 github.com/juju/loggo v0.0.0-20180524022052-584905176618 h1:MK144iBQF9hTSwBW/9eJm034bVoG30IshVm688T2hi8=
15
 github.com/juju/loggo v0.0.0-20180524022052-584905176618 h1:MK144iBQF9hTSwBW/9eJm034bVoG30IshVm688T2hi8=
18
 github.com/juju/loggo v0.0.0-20180524022052-584905176618/go.mod h1:vgyd7OREkbtVEN/8IXZe5Ooef3LQePvuBm9UWj6ZL8U=
16
 github.com/juju/loggo v0.0.0-20180524022052-584905176618/go.mod h1:vgyd7OREkbtVEN/8IXZe5Ooef3LQePvuBm9UWj6ZL8U=
19
 github.com/juju/testing v0.0.0-20180920084828-472a3e8b2073 h1:WQM1NildKThwdP7qWrNAFGzp4ijNLw8RlgENkaI4MJs=
17
 github.com/juju/testing v0.0.0-20180920084828-472a3e8b2073 h1:WQM1NildKThwdP7qWrNAFGzp4ijNLw8RlgENkaI4MJs=
21
 github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI=
19
 github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI=
22
 github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
20
 github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
23
 github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
21
 github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
24
-github.com/kr/pty v1.1.3/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
25
 github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
22
 github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
26
 github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
23
 github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
27
 github.com/pkg/errors v0.8.0 h1:WdK/asTD0HN+q6hsWO3/vpuAkAr+tw6aNJNDFFf0+qw=
24
 github.com/pkg/errors v0.8.0 h1:WdK/asTD0HN+q6hsWO3/vpuAkAr+tw6aNJNDFFf0+qw=
36
 go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
33
 go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
37
 go.uber.org/zap v1.9.1 h1:XCJQEf3W6eZaVwhRBof6ImoYGJSITeKWsyeh3HFu/5o=
34
 go.uber.org/zap v1.9.1 h1:XCJQEf3W6eZaVwhRBof6ImoYGJSITeKWsyeh3HFu/5o=
38
 go.uber.org/zap v1.9.1/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
35
 go.uber.org/zap v1.9.1/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
39
-golang.org/x/net v0.0.0-20180921000356-2f5d2388922f h1:QM2QVxvDoW9PFSPp/zy9FgxJLfaWTZlS61KEPtBwacM=
40
-golang.org/x/net v0.0.0-20180921000356-2f5d2388922f/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
41
-golang.org/x/net v0.0.0-20181005035420-146acd28ed58 h1:otZG8yDCO4LVps5+9bxOeNiCvgmOyt96J3roHTYs7oE=
42
-golang.org/x/net v0.0.0-20181005035420-146acd28ed58/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
36
+golang.org/x/net v0.0.0-20181011144130-49bb7cea24b1 h1:Y/KGZSOdz/2r0WJ9Mkmz6NJBusp0kiNx1Cn82lzJQ6w=
37
+golang.org/x/net v0.0.0-20181011144130-49bb7cea24b1/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
43
 gopkg.in/alecthomas/kingpin.v2 v2.2.6 h1:jMFz6MfLP0/4fUyZle81rXUoxOBFi19VUFKVDOQfozc=
38
 gopkg.in/alecthomas/kingpin.v2 v2.2.6 h1:jMFz6MfLP0/4fUyZle81rXUoxOBFi19VUFKVDOQfozc=
44
 gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
39
 gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
45
 gopkg.in/alexcesaro/statsd.v2 v2.0.0 h1:FXkZSCZIH17vLCO5sO2UucTHsH9pc+17F6pl3JVCwMc=
40
 gopkg.in/alexcesaro/statsd.v2 v2.0.0 h1:FXkZSCZIH17vLCO5sO2UucTHsH9pc+17F6pl3JVCwMc=

+ 6
- 16
proxy/proxy.go Просмотреть файл

93
 		clientPacket := clientConn.(wrappers.PacketReadWriteCloser)
93
 		clientPacket := clientConn.(wrappers.PacketReadWriteCloser)
94
 		serverPacket := serverConn.(wrappers.PacketReadWriteCloser)
94
 		serverPacket := serverConn.(wrappers.PacketReadWriteCloser)
95
 		go p.middlePipe(clientPacket, serverPacket, wait, &opts.ReadHacks)
95
 		go p.middlePipe(clientPacket, serverPacket, wait, &opts.ReadHacks)
96
-		go p.middlePipe(serverPacket, clientPacket, wait, &opts.WriteHacks)
96
+		p.middlePipe(serverPacket, clientPacket, wait, &opts.WriteHacks)
97
 	} else {
97
 	} else {
98
 		clientStream := clientConn.(wrappers.StreamReadWriteCloser)
98
 		clientStream := clientConn.(wrappers.StreamReadWriteCloser)
99
 		serverStream := serverConn.(wrappers.StreamReadWriteCloser)
99
 		serverStream := serverConn.(wrappers.StreamReadWriteCloser)
100
 		go p.directPipe(clientStream, serverStream, wait, p.conf.ReadBufferSize)
100
 		go p.directPipe(clientStream, serverStream, wait, p.conf.ReadBufferSize)
101
-		go p.directPipe(serverStream, clientStream, wait, p.conf.WriteBufferSize)
101
+		p.directPipe(serverStream, clientStream, wait, p.conf.WriteBufferSize)
102
 	}
102
 	}
103
 
103
 
104
 	wait.Wait()
104
 	wait.Wait()
121
 	return packetConn, nil
121
 	return packetConn, nil
122
 }
122
 }
123
 
123
 
124
-func (p *Proxy) middlePipe(src wrappers.PacketReadCloser, dst io.WriteCloser,
125
-	wait *sync.WaitGroup, hacks *mtproto.Hacks) {
126
-	defer func() {
127
-		src.Close() // nolint: errcheck, gosec
128
-		dst.Close() // nolint: errcheck, gosec
129
-		wait.Done()
130
-	}()
124
+func (p *Proxy) middlePipe(src wrappers.PacketReadCloser, dst io.Writer, wait *sync.WaitGroup, hacks *mtproto.Hacks) {
125
+	defer wait.Done()
131
 
126
 
132
 	for {
127
 	for {
133
 		hacks.SimpleAck = false
128
 		hacks.SimpleAck = false
145
 	}
140
 	}
146
 }
141
 }
147
 
142
 
148
-func (p *Proxy) directPipe(src wrappers.StreamReadCloser, dst io.WriteCloser,
149
-	wait *sync.WaitGroup, bufferSize int) {
150
-	defer func() {
151
-		src.Close() // nolint: errcheck, gosec
152
-		dst.Close() // nolint: errcheck, gosec
153
-		wait.Done()
154
-	}()
143
+func (p *Proxy) directPipe(src wrappers.StreamReadCloser, dst io.Writer, wait *sync.WaitGroup, bufferSize int) {
144
+	defer wait.Done()
155
 
145
 
156
 	buffer := make([]byte, bufferSize)
146
 	buffer := make([]byte, bufferSize)
157
 	if _, err := io.CopyBuffer(dst, src, buffer); err != nil {
147
 	if _, err := io.CopyBuffer(dst, src, buffer); err != nil {

+ 20
- 37
wrappers/conn.go Просмотреть файл

38
 	connTimeoutWrite = 2 * time.Minute
38
 	connTimeoutWrite = 2 * time.Minute
39
 )
39
 )
40
 
40
 
41
-type ioResult struct {
42
-	n   int
43
-	err error
44
-}
45
-
46
-type ioFunc func([]byte) (int, error)
47
-
48
 // Conn is a basic wrapper for net.Conn providing the most low-level
41
 // Conn is a basic wrapper for net.Conn providing the most low-level
49
 // logic and management as possible.
42
 // logic and management as possible.
50
 type Conn struct {
43
 type Conn struct {
61
 func (c *Conn) Write(p []byte) (int, error) {
54
 func (c *Conn) Write(p []byte) (int, error) {
62
 	select {
55
 	select {
63
 	case <-c.ctx.Done():
56
 	case <-c.ctx.Done():
57
+		c.Close() // nolint: gosec
64
 		return 0, errors.Annotate(c.ctx.Err(), "Cannot write because context was closed")
58
 		return 0, errors.Annotate(c.ctx.Err(), "Cannot write because context was closed")
65
 	default:
59
 	default:
66
-		n, err := c.doIO(c.conn.Write, p, connTimeoutWrite)
60
+		if err := c.conn.SetWriteDeadline(time.Now().Add(connTimeoutWrite)); err != nil {
61
+			c.Close() // nolint: gosec
62
+			return 0, errors.Annotate(err, "Cannot set write deadline to the socket")
63
+		}
67
 
64
 
65
+		n, err := c.conn.Write(p)
68
 		c.logger.Debugw("Write to stream", "bytes", n, "error", err)
66
 		c.logger.Debugw("Write to stream", "bytes", n, "error", err)
69
 		stats.EgressTraffic(n)
67
 		stats.EgressTraffic(n)
68
+		if err != nil {
69
+			c.Close() // nolint: gosec
70
+		}
70
 
71
 
71
 		return n, err
72
 		return n, err
72
 	}
73
 	}
75
 func (c *Conn) Read(p []byte) (int, error) {
76
 func (c *Conn) Read(p []byte) (int, error) {
76
 	select {
77
 	select {
77
 	case <-c.ctx.Done():
78
 	case <-c.ctx.Done():
79
+		c.Close() // nolint: gosec
78
 		return 0, errors.Annotate(c.ctx.Err(), "Cannot read because context was closed")
80
 		return 0, errors.Annotate(c.ctx.Err(), "Cannot read because context was closed")
79
 	default:
81
 	default:
80
-		n, err := c.doIO(c.conn.Read, p, connTimeoutRead)
82
+		if err := c.conn.SetReadDeadline(time.Now().Add(connTimeoutRead)); err != nil {
83
+			c.Close() // nolint: gosec
84
+			return 0, errors.Annotate(err, "Cannot set read deadline to the socket")
85
+		}
81
 
86
 
87
+		n, err := c.conn.Read(p)
82
 		c.logger.Debugw("Read from stream", "bytes", n, "error", err)
88
 		c.logger.Debugw("Read from stream", "bytes", n, "error", err)
83
 		stats.IngressTraffic(n)
89
 		stats.IngressTraffic(n)
84
-
85
-		return n, err
86
-	}
87
-}
88
-
89
-func (c *Conn) doIO(callback ioFunc, p []byte, timeout time.Duration) (int, error) {
90
-	resChan := make(chan ioResult, 1)
91
-	timer := time.NewTimer(timeout)
92
-
93
-	go func() {
94
-		n, err := callback(p)
95
-		resChan <- ioResult{n: n, err: err}
96
-	}()
97
-
98
-	select {
99
-	case res := <-resChan:
100
-		timer.Stop()
101
-		if res.err != nil {
90
+		if err != nil {
102
 			c.Close() // nolint: gosec
91
 			c.Close() // nolint: gosec
103
 		}
92
 		}
104
-		return res.n, res.err
105
-	case <-c.ctx.Done():
106
-		timer.Stop()
107
-		c.Close() // nolint: gosec
108
-		return 0, errors.Annotate(c.ctx.Err(), "Cannot do IO because context is closed")
109
-	case <-timer.C:
110
-		c.Close() // nolint: gosec
111
-		return 0, errors.Annotate(c.ctx.Err(), "Timeout on IO operation")
93
+
94
+		return n, err
112
 	}
95
 	}
113
 }
96
 }
114
 
97
 
115
 // Close closes underlying net.Conn instance.
98
 // Close closes underlying net.Conn instance.
116
 func (c *Conn) Close() error {
99
 func (c *Conn) Close() error {
117
-	defer c.logger.Debugw("Close connection")
118
-
100
+	c.logger.Debugw("Close connection")
119
 	c.cancel()
101
 	c.cancel()
102
+
120
 	return c.conn.Close()
103
 	return c.conn.Close()
121
 }
104
 }
122
 
105
 

Загрузка…
Отмена
Сохранить