2019-04-19 02:05:02 +08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"flag"
|
|
|
|
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/operation"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/volume_server_pb"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/security"
|
|
|
|
weed_server "github.com/chrislusf/seaweedfs/weed/server"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/storage"
|
|
|
|
"github.com/spf13/viper"
|
2019-04-19 10:22:13 +08:00
|
|
|
"golang.org/x/tools/godoc/util"
|
2019-04-19 02:05:02 +08:00
|
|
|
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
master = flag.String("master", "localhost:9333", "master server host and port")
|
|
|
|
volumeId = flag.Int("volumeId", -1, "a volume id")
|
|
|
|
timeoutSeconds = flag.Int("timeoutSeconds", 0, "disconnect if no activity after these seconds")
|
2019-04-19 10:22:13 +08:00
|
|
|
showTextFile = flag.Bool("showTextFile", false, "display textual file content")
|
2019-04-19 02:05:02 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
weed_server.LoadConfiguration("security", false)
|
|
|
|
grpcDialOption := security.LoadClientTLS(viper.Sub("grpc"), "client")
|
|
|
|
|
|
|
|
vid := storage.VolumeId(*volumeId)
|
|
|
|
|
|
|
|
// find volume location, replication, ttl info
|
|
|
|
lookup, err := operation.Lookup(*master, vid.String())
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Error looking up volume %d: %v", vid, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
volumeServer := lookup.Locations[0].Url
|
|
|
|
log.Printf("volume %d is on volume server %s", vid, volumeServer)
|
|
|
|
|
|
|
|
err = operation.WithVolumeServerClient(volumeServer, grpcDialOption, func(client volume_server_pb.VolumeServerClient) error {
|
|
|
|
|
|
|
|
stream, err := client.VolumeTail(context.Background(), &volume_server_pb.VolumeTailRequest{
|
|
|
|
VolumeId: uint32(vid),
|
|
|
|
SinceNs: 0,
|
|
|
|
DrainingSeconds: uint32(*timeoutSeconds),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
resp, recvErr := stream.Recv()
|
|
|
|
if recvErr != nil {
|
|
|
|
if recvErr == io.EOF {
|
|
|
|
break
|
|
|
|
} else {
|
|
|
|
return recvErr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-19 10:22:13 +08:00
|
|
|
needleHeader := resp.NeedleHeader
|
|
|
|
needleBody := resp.NeedleBody
|
|
|
|
|
|
|
|
if len(needleHeader) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for !resp.IsLastChunk {
|
|
|
|
resp, recvErr = stream.Recv()
|
|
|
|
if recvErr != nil {
|
|
|
|
if recvErr == io.EOF {
|
|
|
|
break
|
|
|
|
} else {
|
|
|
|
return recvErr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
needleBody = append(needleBody, resp.NeedleBody...)
|
|
|
|
}
|
|
|
|
|
|
|
|
n := new(storage.Needle)
|
|
|
|
n.ParseNeedleHeader(needleHeader)
|
|
|
|
n.ReadNeedleBodyBytes(needleBody, storage.CurrentVersion)
|
|
|
|
|
|
|
|
if n.Size == 0 {
|
|
|
|
println("-", n.String())
|
|
|
|
continue
|
|
|
|
} else {
|
|
|
|
println("+", n.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
if *showTextFile {
|
|
|
|
|
|
|
|
data := n.Data
|
|
|
|
if n.IsGzipped() {
|
|
|
|
if data, err = operation.UnGzipData(data); err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if util.IsText(data) {
|
|
|
|
println(string(data))
|
|
|
|
}
|
|
|
|
|
|
|
|
println("-", n.String(), "compressed", n.IsGzipped(), "original size", len(data))
|
|
|
|
}
|
|
|
|
|
2019-04-19 02:05:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Error VolumeTail volume %d: %v", vid, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|