wip: refactory
This commit is contained in:
		| @@ -5,43 +5,61 @@ import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
|  | ||||
| 	"github.com/loveuer/nf/nft/log" | ||||
| 	"github.com/loveuer/esgo2dump/pkg/log" | ||||
|  | ||||
| 	"github.com/loveuer/esgo2dump/internal/opt" | ||||
| 	"github.com/loveuer/esgo2dump/internal/tool" | ||||
| 	"github.com/spf13/cobra" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	rootCommand = &cobra.Command{ | ||||
| 		Use:           "esgo2dump", | ||||
| 		Short:         "esgo2dump is alternative to elasticdump", | ||||
| 		SilenceUsage:  true, | ||||
| 		SilenceErrors: true, | ||||
| 		RunE:          run, | ||||
| 		PersistentPreRun: func(cmd *cobra.Command, args []string) { | ||||
| 			if opt.Cfg.Debug { | ||||
| 				log.SetLogLevel(log.LogLevelDebug) | ||||
| 			} | ||||
| var rootCommand = &cobra.Command{ | ||||
| 	Use:           "esgo2dump", | ||||
| 	Short:         "esgo2dump is alternative to elasticdump", | ||||
| 	SilenceUsage:  true, | ||||
| 	SilenceErrors: true, | ||||
| 	RunE:          run, | ||||
| 	PersistentPreRunE: func(cmd *cobra.Command, args []string) error { | ||||
| 		if opt.Cfg.Debug { | ||||
| 			log.SetLogLevel(log.LogLevelDebug) | ||||
| 		} | ||||
|  | ||||
| 			if opt.Cfg.Args.Version { | ||||
| 				fmt.Printf("esgo2dump version: %s\n", opt.Version) | ||||
| 				os.Exit(0) | ||||
| 			} | ||||
| 		if opt.Cfg.Args.Version { | ||||
| 			fmt.Printf("esgo2dump version: %s\n", opt.Version) | ||||
| 			os.Exit(0) | ||||
| 		} | ||||
|  | ||||
| 		if opt.Cfg.Debug { | ||||
| 			tool.TablePrinter(opt.Cfg) | ||||
| 		} | ||||
|  | ||||
| 		// check args | ||||
| 		if opt.Cfg.Args.Input == "" { | ||||
| 			return cmd.Help() | ||||
| 		} | ||||
|  | ||||
| 		if opt.Cfg.Args.Limit == 0 || opt.Cfg.Args.Limit > 10000 { | ||||
| 			return fmt.Errorf("invalid limit(1 - 10000)") | ||||
| 		} | ||||
|  | ||||
| 		if opt.Cfg.Args.Query != "" && opt.Cfg.Args.QueryFile != "" { | ||||
| 			return fmt.Errorf("cannot specify both query and query_file at the same time") | ||||
| 		} | ||||
|  | ||||
| 		switch opt.Cfg.Args.Type { | ||||
| 		case "data", "mapping", "setting": | ||||
| 		default: | ||||
| 			return fmt.Errorf("unknown type=%s", opt.Cfg.Args.Type) | ||||
| 		} | ||||
|  | ||||
| 		return nil | ||||
| 	}, | ||||
| 	Example: ` | ||||
| esgo2dump -i https://<user>:<password>@<es_node1_host>:<es_node1_port>,<es_node2_host>:<es_node2_port>/some_index?ping=false&sniff=false -o ./data.json | ||||
|  | ||||
| 			if opt.Cfg.Debug { | ||||
| 				tool.TablePrinter(opt.Cfg) | ||||
| 			} | ||||
| 		}, | ||||
| 		Example: ` | ||||
| esgo2dump --input=http://127.0.0.1:9200/some_index --output=./data.json | ||||
|  | ||||
| esgo2dump --input=http://127.0.0.1:9200/some_index --output=http://192.168.1.1:9200/some_index --limit=5000 | ||||
|  | ||||
| esgo2dump --input=http://127.0.0.1:9200/some_index --i-version 6 --output=./data.json | ||||
|  | ||||
| esgo2dump --output=http://127.0.0.1:9200/some_index --o-version 6 --input=./data.json | ||||
|  | ||||
| esgo2dump --input=https://username:password@127.0.0.1:9200/some_index --output=./data.json | ||||
|  | ||||
| esgo2dump --input=http://127.0.0.1:9200/some_index --source='id;name;age;address' --output=./data.json | ||||
| @@ -49,10 +67,7 @@ esgo2dump --input=http://127.0.0.1:9200/some_index --source='id;name;age;address | ||||
| esgo2dump --input=http://127.0.0.1:9200/some_index --output=./data.json --query='{"match": {"name": "some_name"}}' | ||||
|  | ||||
| esgo2dump --input=http://127.0.0.1:9200/some_index --output=./data.json --query_file=my_queries.json`, | ||||
| 	} | ||||
|  | ||||
| 	es_iversion, es_oversion string | ||||
| ) | ||||
| } | ||||
|  | ||||
| func init() { | ||||
| 	rootCommand.PersistentFlags().BoolVar(&opt.Cfg.Debug, "debug", false, "") | ||||
| @@ -63,16 +78,15 @@ func init() { | ||||
| 	rootCommand.Flags().IntVar(&opt.Cfg.Args.Timeout, "timeout", 30, "max timeout seconds per operation with limit") | ||||
| 	rootCommand.Flags().StringVarP(&opt.Cfg.Args.Input, "input", "i", "", "*required: input file or es url (example :data.json / http://127.0.0.1:9200/my_index)") | ||||
| 	rootCommand.Flags().StringVarP(&opt.Cfg.Args.Output, "output", "o", "output.json", "") | ||||
| 	rootCommand.Flags().StringVar(&es_iversion, "i-version", "7", "input(es) version") | ||||
| 	rootCommand.Flags().StringVar(&es_oversion, "o-version", "7", "output(es) version") | ||||
| 	rootCommand.Flags().StringVarP(&opt.Cfg.Args.Type, "type", "t", "data", "data/mapping/setting") | ||||
| 	rootCommand.Flags().StringVar(&opt.Cfg.Args.Source, "source", "", "query source, use ';' to separate") | ||||
| 	rootCommand.Flags().StringVar(&opt.Cfg.Args.Sort, "sort", "", "sort, <field>:<direction> format, for example: time:desc or name:asc") | ||||
| 	rootCommand.Flags().StringVar(&opt.Cfg.Args.Field, "field", "", "query include field, use ',' to separate") | ||||
| 	rootCommand.Flags().StringVar(&opt.Cfg.Args.Sort, "sort", "", "sort, <field>:<direction> format, for example: time:desc or name:asc, user ',' to separate") | ||||
| 	rootCommand.Flags().StringVar(&opt.Cfg.Args.Query, "query", "", `query dsl, example: {"bool":{"must":[{"term":{"name":{"value":"some_name"}}}],"must_not":[{"range":{"age":{"gte":18,"lt":60}}}]}}`) | ||||
| 	rootCommand.Flags().StringVar(&opt.Cfg.Args.QueryFile, "query_file", "", `query json file (will execute line by line)`) | ||||
| 	rootCommand.Flags().IntVar(&opt.Cfg.Args.Limit, "limit", 100, "") | ||||
| 	rootCommand.Flags().IntVar(&opt.Cfg.Args.Max, "max", 0, "max dump records") | ||||
| } | ||||
|  | ||||
| func Start(ctx context.Context) error { | ||||
| func Run(ctx context.Context) error { | ||||
| 	return rootCommand.ExecuteContext(ctx) | ||||
| } | ||||
|   | ||||
| @@ -4,302 +4,222 @@ import ( | ||||
| 	"bufio" | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/loveuer/esgo2dump/model" | ||||
| 	"github.com/loveuer/nf/nft/log" | ||||
|  | ||||
| 	"github.com/loveuer/esgo2dump/internal/interfaces" | ||||
| 	elastic "github.com/elastic/go-elasticsearch/v7" | ||||
| 	"github.com/go-resty/resty/v2" | ||||
| 	"github.com/loveuer/esgo2dump/internal/opt" | ||||
| 	"github.com/loveuer/esgo2dump/internal/xes" | ||||
| 	"github.com/loveuer/esgo2dump/internal/tool" | ||||
| 	"github.com/loveuer/esgo2dump/internal/xfile" | ||||
| 	"github.com/loveuer/esgo2dump/pkg/log" | ||||
| 	"github.com/loveuer/esgo2dump/pkg/model" | ||||
| 	"github.com/loveuer/esgo2dump/xes/es7" | ||||
| 	"github.com/samber/lo" | ||||
| 	"github.com/spf13/cobra" | ||||
| ) | ||||
|  | ||||
| func check(cmd *cobra.Command) error { | ||||
| 	if opt.Cfg.Args.Input == "" { | ||||
| 		return cmd.Help() | ||||
| 		// return fmt.Errorf("must specify input(example: data.json/http://127.0.0.1:9200/my_index)") | ||||
| func newIO(ctx context.Context, uri string, ioType model.IOType) (model.IO[map[string]any], error) { | ||||
| 	type Version struct { | ||||
| 		Name    string | ||||
| 		Version struct { | ||||
| 			Number string `json:"number"` | ||||
| 		} `json:"version"` | ||||
| 	} | ||||
|  | ||||
| 	if opt.Cfg.Args.Limit == 0 || opt.Cfg.Args.Limit > 10000 { | ||||
| 		return fmt.Errorf("invalid limit(1 - 10000)") | ||||
| 	var ( | ||||
| 		err    error | ||||
| 		target *url.URL | ||||
| 		rr     *resty.Response | ||||
| 		v      Version | ||||
| 	) | ||||
|  | ||||
| 	if target, err = url.Parse(uri); err != nil { | ||||
| 		log.Debug("parse uri failed, type = %s, uri = %s, err = %s", ioType, uri, err.Error()) | ||||
| 		return xfile.NewClient(uri, ioType) | ||||
| 	} | ||||
|  | ||||
| 	if opt.Cfg.Args.Query != "" && opt.Cfg.Args.QueryFile != "" { | ||||
| 		return fmt.Errorf("cannot specify both query and query_file at the same time") | ||||
| 	if err = tool.ValidScheme(target.Scheme); err != nil { | ||||
| 		log.Debug("uri scheme check failed, type = %s, uri = %s", ioType, uri) | ||||
| 		return xfile.NewClient(uri, ioType) | ||||
| 	} | ||||
|  | ||||
| 	switch opt.Cfg.Args.Type { | ||||
| 	case "data", "mapping", "setting": | ||||
| 	// elastic uri | ||||
| 	index := strings.TrimPrefix(target.Path, "/") | ||||
| 	if index == "" { | ||||
| 		return nil, fmt.Errorf("uri invalid without index(path)") | ||||
| 	} | ||||
|  | ||||
| 	log.Debug("%s uri es index = %s", ioType, index) | ||||
|  | ||||
| 	versionURL := fmt.Sprintf("%s://%s", target.Scheme, strings.Split(target.Host, ",")[0]) | ||||
| 	log.Debug("%s version url = %s", ioType, versionURL) | ||||
| 	if rr, err = opt.HttpClient.R().Get(versionURL); err != nil { | ||||
| 		log.Debug("get uri es version failed, type = %s, uri = %s, version_url = %s, err = %s", ioType, uri, versionURL, err.Error()) | ||||
| 	} | ||||
|  | ||||
| 	if err = json.Unmarshal(rr.Body(), &v); err != nil { | ||||
| 		log.Debug("decode uri es version failed, type = %s, uri = %s, version_url = %s, err = %s", ioType, uri, versionURL, err.Error()) | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	log.Debug("%s uri es version = %s", ioType, v.Version.Number) | ||||
|  | ||||
| 	mainVersion := strings.Split(v.Version.Number, ".")[0] | ||||
| 	switch mainVersion { | ||||
| 	case "8": | ||||
| 	case "7": | ||||
| 		var client *elastic.Client | ||||
| 		if client, err = es7.NewClient(ctx, uri); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		return es7.NewStreamer(ctx, client, index) | ||||
| 	case "6": | ||||
| 	default: | ||||
| 		return fmt.Errorf("unknown type=%s", opt.Cfg.Args.Type) | ||||
| 		return nil, fmt.Errorf("es version not supported yet: %s", mainVersion) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| 	return nil, nil | ||||
| } | ||||
|  | ||||
| func run(cmd *cobra.Command, args []string) error { | ||||
| 	var ( | ||||
| 		err error | ||||
| 		ioi interfaces.DumpIO | ||||
| 		ioo interfaces.DumpIO | ||||
| 		err    error | ||||
| 		input  model.IO[map[string]any] | ||||
| 		output model.IO[map[string]any] | ||||
| 	) | ||||
|  | ||||
| 	if err = check(cmd); err != nil { | ||||
| 	if input, err = newIO(cmd.Context(), opt.Cfg.Args.Input, model.Input); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if ioi, err = newIO(opt.Cfg.Args.Input, interfaces.IOInput, es_iversion); err != nil { | ||||
| 	if output, err = newIO(cmd.Context(), opt.Cfg.Args.Output, model.Output); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	log.Debug("init: new input io success!") | ||||
|  | ||||
| 	if ioo, err = newIO(opt.Cfg.Args.Output, interfaces.IOOutput, es_oversion); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	log.Debug("init: new output io success!") | ||||
|  | ||||
| 	defer func() { | ||||
| 		_ = ioi.Close() | ||||
| 		_ = ioo.Close() | ||||
| 	go func() { | ||||
| 		<-cmd.Context().Done() | ||||
| 		os.Exit(1) | ||||
| 	}() | ||||
|  | ||||
| 	if (opt.Cfg.Args.Query != "" || opt.Cfg.Args.QueryFile != "") && ioi.IsFile() { | ||||
| 		return fmt.Errorf("with file input, query or query_file can't be supported") | ||||
| 	} | ||||
|  | ||||
| 	if (opt.Cfg.Args.Source != "") && ioi.IsFile() { | ||||
| 		return fmt.Errorf("with file input, source can't be supported") | ||||
| 	} | ||||
|  | ||||
| 	switch opt.Cfg.Args.Type { | ||||
| 	case "data": | ||||
| 		if err = executeData(cmd.Context(), ioi, ioo); err != nil { | ||||
| 	if opt.Cfg.Args.QueryFile != "" { | ||||
| 		// query file | ||||
| 		var ( | ||||
| 			items []map[string]any | ||||
| 			qf    *os.File | ||||
| 			// wrote count | ||||
| 			wc int | ||||
| 		) | ||||
| 		if qf, err = os.Open(opt.Cfg.Args.QueryFile); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		log.Info("Dump: write data succeed!!!") | ||||
| 		scanner := bufio.NewScanner(qf) | ||||
| 		// query count | ||||
| 		qc := 0 | ||||
| 		for scanner.Scan() { | ||||
| 			qc++ | ||||
| 			qm := make(map[string]any) | ||||
| 			if err = json.Unmarshal(scanner.Bytes(), &qm); err != nil { | ||||
| 				return err | ||||
| 			} | ||||
|  | ||||
| 		return nil | ||||
| 	case "mapping": | ||||
| 		var mapping map[string]any | ||||
| 		if mapping, err = ioi.ReadMapping(cmd.Context()); err != nil { | ||||
| 			return err | ||||
| 			for { | ||||
| 				if items, err = input.ReadData( | ||||
| 					opt.Cfg.Args.Limit, | ||||
| 					qm, | ||||
| 					lo.Filter(strings.Split(opt.Cfg.Args.Field, ","), func(x string, _ int) bool { return x != "" }), | ||||
| 					lo.Filter(strings.Split(opt.Cfg.Args.Sort, ","), func(x string, _ int) bool { return x != "" }), | ||||
| 				); err != nil { | ||||
| 					return err | ||||
| 				} | ||||
|  | ||||
| 				if len(items) == 0 { | ||||
| 					break | ||||
| 				} | ||||
|  | ||||
| 				if wc, err = output.WriteData(items); err != nil { | ||||
| 					return err | ||||
| 				} | ||||
|  | ||||
| 				if wc != len(items) { | ||||
| 					return fmt.Errorf("got items %d, but wrote %d", len(items), wc) | ||||
| 				} | ||||
|  | ||||
| 				log.Info("Dump: query_file[%06d] dump success = %d", qc, wc) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if err = ioo.WriteMapping(cmd.Context(), mapping); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		log.Info("Dump: write mapping succeed!!!") | ||||
|  | ||||
| 		return nil | ||||
| 	case "setting": | ||||
| 		var setting map[string]any | ||||
| 		if setting, err = ioi.ReadSetting(cmd.Context()); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err = ioo.WriteSetting(cmd.Context(), setting); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		log.Info("Dump: write setting succeed!!!") | ||||
|  | ||||
| 		return nil | ||||
| 	default: | ||||
| 		return fmt.Errorf("unknown type=%s", opt.Cfg.Args.Type) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func executeData(ctx context.Context, input, output interfaces.DumpIO) error { | ||||
| 	var ( | ||||
| 		err     error | ||||
| 		queries = make([]map[string]any, 0) | ||||
| 		sources = make([]string, 0) | ||||
| 	) | ||||
|  | ||||
| 	if opt.Cfg.Args.Source != "" { | ||||
| 		sources = lo.Map(strings.Split(opt.Cfg.Args.Source, ";"), func(item string, idx int) string { | ||||
| 			return strings.TrimSpace(item) | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	if opt.Cfg.Args.Query != "" { | ||||
| 		query := make(map[string]any) | ||||
| 		if err = json.Unmarshal([]byte(opt.Cfg.Args.Query), &query); err != nil { | ||||
| 			return fmt.Errorf("invalid query err=%v", err) | ||||
| 		var ( | ||||
| 			items []map[string]any | ||||
| 			qm    = make(map[string]any) | ||||
| 			wc    int | ||||
| 		) | ||||
|  | ||||
| 		if err = json.Unmarshal([]byte(opt.Cfg.Args.Query), &qm); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		queries = append(queries, query) | ||||
| 	} | ||||
|  | ||||
| 	if opt.Cfg.Args.QueryFile != "" { | ||||
| 		var qf *os.File | ||||
|  | ||||
| 		if qf, err = os.Open(opt.Cfg.Args.QueryFile); err != nil { | ||||
| 			return fmt.Errorf("open query_file err=%v", err) | ||||
| 		} | ||||
|  | ||||
| 		defer func() { | ||||
| 			_ = qf.Close() | ||||
| 		}() | ||||
|  | ||||
| 		scanner := bufio.NewScanner(qf) | ||||
| 		scanner.Buffer(make([]byte, 1*1024*1024), 5*1024*1024) | ||||
| 		lineCount := 1 | ||||
| 		for scanner.Scan() { | ||||
| 			line := scanner.Text() | ||||
| 			oq := make(map[string]any) | ||||
| 			if err = json.Unmarshal([]byte(line), &oq); err != nil { | ||||
| 				return fmt.Errorf("query file line=%d invalid err=%v", lineCount, err) | ||||
| 		for { | ||||
| 			if items, err = input.ReadData( | ||||
| 				opt.Cfg.Args.Limit, | ||||
| 				qm, | ||||
| 				lo.Filter(strings.Split(opt.Cfg.Args.Field, ","), func(x string, _ int) bool { return x != "" }), | ||||
| 				lo.Filter(strings.Split(opt.Cfg.Args.Sort, ","), func(x string, _ int) bool { return x != "" }), | ||||
| 			); err != nil { | ||||
| 				return err | ||||
| 			} | ||||
|  | ||||
| 			queries = append(queries, oq) | ||||
|  | ||||
| 			if len(queries) > 10000 { | ||||
| 				return fmt.Errorf("query_file support max lines=%d", 10000) | ||||
| 			if len(items) == 0 { | ||||
| 				break | ||||
| 			} | ||||
|  | ||||
| 			lineCount++ | ||||
| 			if wc, err = output.WriteData(items); err != nil { | ||||
| 				return err | ||||
| 			} | ||||
|  | ||||
| 			if wc != len(items) { | ||||
| 				return fmt.Errorf("got items %d, but wrote %d", len(items), wc) | ||||
| 			} | ||||
|  | ||||
| 			log.Info("Dump: query dump success = %d", wc) | ||||
| 		} | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	if len(queries) == 0 { | ||||
| 		queries = append(queries, nil) | ||||
| 	} | ||||
|  | ||||
| 	var ( | ||||
| 		ok   bool | ||||
| 		docs []*model.ESSource | ||||
| 		dch  <-chan []*model.ESSource | ||||
| 		ech  <-chan error | ||||
|  | ||||
| 		e2ch = make(chan error) | ||||
| 		wch  = make(chan []*model.ESSource) | ||||
| 		wg   = sync.WaitGroup{} | ||||
| 		items []map[string]any | ||||
| 		wc    int | ||||
| 	) | ||||
|  | ||||
| 	wg.Add(1) | ||||
| 	go func() { | ||||
| 		if err = output.WriteData(ctx, wch); err != nil { | ||||
| 			log.Fatal("Dump: write data err: %s", err.Error()) | ||||
| 	for { | ||||
| 		if items, err = input.ReadData( | ||||
| 			opt.Cfg.Args.Limit, | ||||
| 			nil, | ||||
| 			lo.Filter(strings.Split(opt.Cfg.Args.Field, ","), func(x string, _ int) bool { return x != "" }), | ||||
| 			lo.Filter(strings.Split(opt.Cfg.Args.Sort, ","), func(x string, _ int) bool { return x != "" }), | ||||
| 		); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		wg.Done() | ||||
| 	}() | ||||
|  | ||||
| 	log.Info("Query: got queries=%d", len(queries)) | ||||
|  | ||||
| Loop: | ||||
| 	for queryIdx, query := range queries { | ||||
| 		bs, _ := json.Marshal(query) | ||||
|  | ||||
| 		log.Debug("Query[%d]: %s", queryIdx, string(bs)) | ||||
|  | ||||
| 		dch, ech = input.ReadData(ctx, opt.Cfg.Args.Limit, query, sources, []string{opt.Cfg.Args.Sort}) | ||||
|  | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-ctx.Done(): | ||||
| 				return ctx.Err() | ||||
| 			case err, ok = <-ech: | ||||
| 				if !ok { | ||||
| 					log.Debug("pipe: read io closed") | ||||
| 					continue Loop | ||||
| 				} | ||||
| 				log.Debug("pipe: got err from read io, err = %s", err.Error()) | ||||
| 				return err | ||||
| 			case err, ok = <-e2ch: | ||||
| 				if !ok { | ||||
| 					log.Debug("pipe: write io closed") | ||||
| 					continue Loop | ||||
| 				} | ||||
| 				log.Debug("pipe: got err from write io, err = %s", err.Error()) | ||||
| 				return err | ||||
| 			case docs, ok = <-dch: | ||||
| 				if !ok || len(docs) == 0 { | ||||
| 					continue Loop | ||||
| 				} | ||||
|  | ||||
| 				log.Debug("pipe: got %d docs from read io", len(docs)) | ||||
|  | ||||
| 				wch <- docs | ||||
| 			} | ||||
| 		if len(items) == 0 { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		if wc, err = output.WriteData(items); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if wc != len(items) { | ||||
| 			return fmt.Errorf("got items %d, but wrote %d", len(items), wc) | ||||
| 		} | ||||
|  | ||||
| 		log.Info("Dump: query dump success = %d", wc) | ||||
| 	} | ||||
|  | ||||
| 	close(wch) | ||||
|  | ||||
| 	log.Debug("pipe: wait for all io closed") | ||||
| 	wg.Wait() | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func newIO(source string, ioType interfaces.IO, esv string) (interfaces.DumpIO, error) { | ||||
| 	var ( | ||||
| 		err  error | ||||
| 		iurl *url.URL | ||||
| 		file *os.File | ||||
| 		qm   = make(map[string]any) | ||||
| 	) | ||||
|  | ||||
| 	log.Debug("action=%s, type=%s, source=%s, es_version=%s", "new_io", ioType.Code(), source, esv) | ||||
|  | ||||
| 	if iurl, err = url.Parse(source); err != nil { | ||||
| 		log.Debug("action=%s, type=%s, source=%s, err=%s", "new_io url parse err", ioType.Code(), source, err.Error()) | ||||
| 		goto ClientByFile | ||||
| 	} | ||||
|  | ||||
| 	if !(iurl.Scheme == "http" || iurl.Scheme == "https") { | ||||
| 		log.Debug("action=%s, type=%s, source=%s, scheme=%s", "new_io url scheme error", ioType.Code(), source, iurl.Scheme) | ||||
| 		goto ClientByFile | ||||
| 	} | ||||
|  | ||||
| 	if iurl.Host == "" { | ||||
| 		log.Debug("action=%s, type=%s, source=%s", "new_io url host empty", ioType.Code(), source) | ||||
| 		goto ClientByFile | ||||
| 	} | ||||
|  | ||||
| 	if ioType == interfaces.IOInput && opt.Cfg.Args.Query != "" { | ||||
| 		if err = json.Unmarshal([]byte(opt.Cfg.Args.Query), &qm); err != nil { | ||||
| 			log.Debug("action=%s, type=%s, source=%s, query=%s", "new_io query string invalid", ioType.Code(), source, opt.Cfg.Args.Query) | ||||
| 			return nil, fmt.Errorf("invalid query err=%v", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	switch esv { | ||||
| 	case "7": | ||||
| 		return xes.NewClient(source, ioType) | ||||
| 	case "6": | ||||
| 		return xes.NewClientV6(iurl, ioType) | ||||
| 	case "8": | ||||
| 		return nil, errors.New("es version 8 coming soon") | ||||
| 	default: | ||||
| 		return nil, fmt.Errorf("unknown es version=%s", esv) | ||||
| 	} | ||||
|  | ||||
| ClientByFile: | ||||
| 	if ioType == interfaces.IOOutput { | ||||
| 		if _, err = os.Stat(source); !os.IsNotExist(err) { | ||||
| 			return nil, fmt.Errorf("output_file=%s already exist", source) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if file, err = os.OpenFile(source, os.O_CREATE|os.O_RDWR, 0o644); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return xfile.NewClient(file, ioType) | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user