2017-03-09 02:03:47 +08:00
|
|
|
// Copyright 2017 fatedier, fatedier@gmail.com
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2017-06-06 18:48:40 +08:00
|
|
|
package io
|
2017-03-09 02:03:47 +08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
2017-06-06 18:48:40 +08:00
|
|
|
"sync"
|
2017-03-09 02:03:47 +08:00
|
|
|
|
|
|
|
"github.com/fatedier/frp/utils/crypto"
|
2017-06-06 18:48:40 +08:00
|
|
|
"github.com/fatedier/frp/utils/pool"
|
2017-03-09 02:03:47 +08:00
|
|
|
)
|
|
|
|
|
2017-06-06 18:48:40 +08:00
|
|
|
// Join two io.ReadWriteCloser and do some operations.
|
|
|
|
func Join(c1 io.ReadWriteCloser, c2 io.ReadWriteCloser) (inCount int64, outCount int64) {
|
|
|
|
var wait sync.WaitGroup
|
|
|
|
pipe := func(to io.ReadWriteCloser, from io.ReadWriteCloser, count *int64) {
|
|
|
|
defer to.Close()
|
|
|
|
defer from.Close()
|
|
|
|
defer wait.Done()
|
|
|
|
|
|
|
|
buf := pool.GetBuf(16 * 1024)
|
|
|
|
defer pool.PutBuf(buf)
|
|
|
|
*count, _ = io.CopyBuffer(to, from, buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
wait.Add(2)
|
|
|
|
go pipe(c1, c2, &inCount)
|
|
|
|
go pipe(c2, c1, &outCount)
|
|
|
|
wait.Wait()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-10 02:01:17 +08:00
|
|
|
func WithEncryption(rwc io.ReadWriteCloser, key []byte) (io.ReadWriteCloser, error) {
|
|
|
|
w, err := crypto.NewWriter(rwc, key)
|
2017-03-09 02:03:47 +08:00
|
|
|
if err != nil {
|
2017-03-10 02:01:17 +08:00
|
|
|
return nil, err
|
2017-03-09 02:03:47 +08:00
|
|
|
}
|
2017-06-05 23:52:24 +08:00
|
|
|
return WrapReadWriteCloser(crypto.NewReader(rwc, key), w, func() error {
|
|
|
|
return rwc.Close()
|
|
|
|
}), nil
|
2017-03-09 02:03:47 +08:00
|
|
|
}
|
|
|
|
|
2017-03-10 02:01:17 +08:00
|
|
|
func WithCompression(rwc io.ReadWriteCloser) io.ReadWriteCloser {
|
2017-06-08 00:57:33 +08:00
|
|
|
sr := pool.GetSnappyReader(rwc)
|
|
|
|
sw := pool.GetSnappyWriter(rwc)
|
|
|
|
return WrapReadWriteCloser(sr, sw, func() error {
|
|
|
|
err := rwc.Close()
|
|
|
|
pool.PutSnappyReader(sr)
|
|
|
|
pool.PutSnappyWriter(sw)
|
|
|
|
return err
|
2017-06-05 23:52:24 +08:00
|
|
|
})
|
2017-03-09 02:03:47 +08:00
|
|
|
}
|
|
|
|
|
2017-06-05 23:52:24 +08:00
|
|
|
type ReadWriteCloser struct {
|
|
|
|
r io.Reader
|
|
|
|
w io.Writer
|
|
|
|
closeFn func() error
|
2017-06-08 00:57:33 +08:00
|
|
|
|
|
|
|
closed bool
|
|
|
|
mu sync.Mutex
|
2017-03-09 02:03:47 +08:00
|
|
|
}
|
|
|
|
|
2017-06-08 00:57:33 +08:00
|
|
|
// closeFn will be called only once
|
2017-06-05 23:52:24 +08:00
|
|
|
func WrapReadWriteCloser(r io.Reader, w io.Writer, closeFn func() error) io.ReadWriteCloser {
|
|
|
|
return &ReadWriteCloser{
|
|
|
|
r: r,
|
|
|
|
w: w,
|
|
|
|
closeFn: closeFn,
|
2017-06-08 00:57:33 +08:00
|
|
|
closed: false,
|
2017-06-05 23:52:24 +08:00
|
|
|
}
|
2017-03-09 02:03:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (rwc *ReadWriteCloser) Read(p []byte) (n int, err error) {
|
|
|
|
return rwc.r.Read(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rwc *ReadWriteCloser) Write(p []byte) (n int, err error) {
|
|
|
|
return rwc.w.Write(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rwc *ReadWriteCloser) Close() (errRet error) {
|
2017-06-08 00:57:33 +08:00
|
|
|
rwc.mu.Lock()
|
|
|
|
if rwc.closed {
|
|
|
|
rwc.mu.Unlock()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
rwc.closed = true
|
|
|
|
rwc.mu.Unlock()
|
|
|
|
|
2017-03-09 02:03:47 +08:00
|
|
|
var err error
|
|
|
|
if rc, ok := rwc.r.(io.Closer); ok {
|
|
|
|
err = rc.Close()
|
|
|
|
if err != nil {
|
|
|
|
errRet = err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if wc, ok := rwc.w.(io.Closer); ok {
|
|
|
|
err = wc.Close()
|
|
|
|
if err != nil {
|
|
|
|
errRet = err
|
|
|
|
}
|
|
|
|
}
|
2017-06-05 23:52:24 +08:00
|
|
|
|
|
|
|
if rwc.closeFn != nil {
|
|
|
|
err = rwc.closeFn()
|
|
|
|
if err != nil {
|
|
|
|
errRet = err
|
|
|
|
}
|
|
|
|
}
|
2017-03-09 02:03:47 +08:00
|
|
|
return
|
|
|
|
}
|