mirror of
https://github.com/minio/mc.git
synced 2025-11-10 13:42:32 +03:00
210 lines
5.2 KiB
Go
210 lines
5.2 KiB
Go
/*
|
|
* MinIO Client (C) 2016 MinIO, Inc.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"strings"
|
|
"sync"
|
|
"syscall"
|
|
|
|
humanize "github.com/dustin/go-humanize"
|
|
"github.com/fatih/color"
|
|
"github.com/minio/cli"
|
|
json "github.com/minio/mc/pkg/colorjson"
|
|
"github.com/minio/mc/pkg/console"
|
|
"github.com/minio/mc/pkg/probe"
|
|
)
|
|
|
|
var (
|
|
watchFlags = []cli.Flag{
|
|
cli.StringFlag{
|
|
Name: "events",
|
|
Value: "put,delete,get",
|
|
Usage: "filter specific types of events; defaults to all events by default",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "prefix",
|
|
Usage: "filter events for a prefix",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "suffix",
|
|
Usage: "filter events for a suffix",
|
|
},
|
|
cli.BoolFlag{
|
|
Name: "recursive",
|
|
Usage: "recursively watch for events",
|
|
},
|
|
}
|
|
)
|
|
|
|
var watchCmd = cli.Command{
|
|
Name: "watch",
|
|
Usage: "listen for object notification events",
|
|
Action: mainWatch,
|
|
Before: setGlobalsFromContext,
|
|
Flags: append(watchFlags, globalFlags...),
|
|
CustomHelpTemplate: `NAME:
|
|
{{.HelpName}} - {{.Usage}}
|
|
|
|
USAGE:
|
|
{{.HelpName}} [FLAGS] PATH
|
|
{{if .VisibleFlags}}
|
|
FLAGS:
|
|
{{range .VisibleFlags}}{{.}}
|
|
{{end}}{{end}}
|
|
EXAMPLES:
|
|
1. Watch new S3 operations on a MinIO server
|
|
$ {{.HelpName}} play/testbucket
|
|
|
|
2. Watch new events for a specific prefix "output/" on MinIO server.
|
|
$ {{.HelpName}} --prefix "output/" play/testbucket
|
|
|
|
3. Watch new events for a specific suffix ".jpg" on MinIO server.
|
|
$ {{.HelpName}} --suffix ".jpg" play/testbucket
|
|
|
|
4. Watch new events on a specific prefix and suffix on MinIO server.
|
|
$ {{.HelpName}} --suffix ".jpg" --prefix "photos/" play/testbucket
|
|
|
|
5. Watch for events on local directory.
|
|
$ {{.HelpName}} /usr/share
|
|
`,
|
|
}
|
|
|
|
// checkWatchSyntax - validate all the passed arguments
|
|
func checkWatchSyntax(ctx *cli.Context) {
|
|
if len(ctx.Args()) != 1 {
|
|
cli.ShowCommandHelpAndExit(ctx, "watch", 1) // last argument is exit code
|
|
}
|
|
}
|
|
|
|
// watchMessage container to hold one event notification
|
|
type watchMessage struct {
|
|
Status string `json:"status"`
|
|
Event struct {
|
|
Time string `json:"time"`
|
|
Size int64 `json:"size"`
|
|
Path string `json:"path"`
|
|
Type EventType `json:"type"`
|
|
} `json:"events"`
|
|
Source struct {
|
|
Host string `json:"host,omitempty"`
|
|
Port string `json:"port,omitempty"`
|
|
UserAgent string `json:"userAgent,omitempty"`
|
|
} `json:"source,omitempty"`
|
|
}
|
|
|
|
func (u watchMessage) JSON() string {
|
|
u.Status = "success"
|
|
watchMessageJSONBytes, e := json.MarshalIndent(u, "", " ")
|
|
fatalIf(probe.NewError(e), "Unable to marshal into JSON.")
|
|
return string(watchMessageJSONBytes)
|
|
}
|
|
|
|
func (u watchMessage) String() string {
|
|
msg := console.Colorize("Time", fmt.Sprintf("[%s] ", u.Event.Time))
|
|
if u.Event.Type == EventCreate {
|
|
msg += console.Colorize("Size", fmt.Sprintf("%6s ", humanize.IBytes(uint64(u.Event.Size))))
|
|
} else {
|
|
msg += fmt.Sprintf("%6s ", "")
|
|
}
|
|
msg += console.Colorize("EventType", fmt.Sprintf("%s ", u.Event.Type))
|
|
msg += console.Colorize("ObjectName", u.Event.Path)
|
|
return msg
|
|
}
|
|
|
|
func mainWatch(ctx *cli.Context) error {
|
|
console.SetColor("Time", color.New(color.FgGreen))
|
|
console.SetColor("Size", color.New(color.FgYellow))
|
|
console.SetColor("EventType", color.New(color.FgCyan, color.Bold))
|
|
console.SetColor("ObjectName", color.New(color.Bold))
|
|
|
|
checkWatchSyntax(ctx)
|
|
|
|
args := ctx.Args()
|
|
path := args[0]
|
|
|
|
prefix := ctx.String("prefix")
|
|
suffix := ctx.String("suffix")
|
|
events := strings.Split(ctx.String("events"), ",")
|
|
recursive := ctx.Bool("recursive")
|
|
|
|
s3Client, pErr := newClient(path)
|
|
if pErr != nil {
|
|
fatalIf(pErr.Trace(), "Cannot parse the provided url.")
|
|
}
|
|
|
|
params := watchParams{
|
|
recursive: recursive,
|
|
events: events,
|
|
prefix: prefix,
|
|
suffix: suffix,
|
|
}
|
|
|
|
// Start watching on events
|
|
wo, err := s3Client.Watch(params)
|
|
fatalIf(err, "Cannot watch on the specified bucket.")
|
|
|
|
trapCh := signalTrap(os.Interrupt, syscall.SIGTERM)
|
|
|
|
// Initialize.. waitgroup to track the go-routine.
|
|
wg := sync.WaitGroup{}
|
|
|
|
// Increment wait group to wait subsequent routine.
|
|
wg.Add(1)
|
|
|
|
// Start routine to watching on events.
|
|
go func() {
|
|
defer wg.Done()
|
|
|
|
// Wait for all events.
|
|
for {
|
|
select {
|
|
case <-trapCh:
|
|
// Signal received we are done.
|
|
close(wo.doneChan)
|
|
return
|
|
case event, ok := <-wo.Events():
|
|
if !ok {
|
|
return
|
|
}
|
|
msg := watchMessage{}
|
|
msg.Event.Path = event.Path
|
|
msg.Event.Size = event.Size
|
|
msg.Event.Time = event.Time
|
|
msg.Event.Type = event.Type
|
|
msg.Source.Host = event.Host
|
|
msg.Source.Port = event.Port
|
|
msg.Source.UserAgent = event.UserAgent
|
|
printMsg(msg)
|
|
case err, ok := <-wo.Errors():
|
|
if !ok {
|
|
return
|
|
}
|
|
errorIf(err, "Unable to watch for events.")
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
// Wait on the routine to be finished or exit.
|
|
wg.Wait()
|
|
|
|
return nil
|
|
}
|