package main import ( "context" "errors" "fmt" "io/ioutil" "math" "os" "path/filepath" "strconv" "strings" "time" "github.com/thrasher-corp/gocryptotrader/common" "github.com/thrasher-corp/gocryptotrader/currency" "github.com/thrasher-corp/gocryptotrader/gctrpc" "github.com/urfave/cli/v2" "google.golang.org/grpc" ) var startTime, endTime, order string var limit int var getInfoCommand = &cli.Command{ Name: "getinfo", Usage: "gets GoCryptoTrader info", Action: getInfo, } func getInfo(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetInfo(context.Background(), &gctrpc.GetInfoRequest{}, ) if err != nil { return err } jsonOutput(result) return nil } var getSubsystemsCommand = &cli.Command{ Name: "getsubsystems", Usage: "gets GoCryptoTrader subsystems and their status", Action: getSubsystems, } func getSubsystems(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetSubsystems(context.Background(), &gctrpc.GetSubsystemsRequest{}, ) if err != nil { return err } jsonOutput(result) return nil } var enableSubsystemCommand = &cli.Command{ Name: "enablesubsystem", Usage: "enables an engine subsystem", ArgsUsage: "", Action: enableSubsystem, Flags: []cli.Flag{ &cli.StringFlag{ Name: "subsystem", Usage: "the subsystem to enable", }, }, } func enableSubsystem(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "enablesubsystem") } var subsystemName string if c.IsSet("subsystem") { subsystemName = c.String("subsystem") } else { subsystemName = c.Args().First() } if subsystemName == "" { return errors.New("invalid subsystem supplied") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.EnableSubsystem(context.Background(), &gctrpc.GenericSubsystemRequest{ Subsystem: subsystemName, }, ) if err != nil { return err } jsonOutput(result) return nil } var disableSubsystemCommand = &cli.Command{ Name: "disablesubsystem", Usage: "disables an engine subsystem", ArgsUsage: "", Action: disableSubsystem, Flags: []cli.Flag{ &cli.StringFlag{ Name: "subsystem", Usage: "the subsystem to disable", }, }, } func disableSubsystem(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "disablesubsystem") } var subsystemName string if c.IsSet("subsystem") { subsystemName = c.String("subsystem") } else { subsystemName = c.Args().First() } if subsystemName == "" { return errors.New("invalid subsystem supplied") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.DisableSubsystem(context.Background(), &gctrpc.GenericSubsystemRequest{ Subsystem: subsystemName, }, ) if err != nil { return err } jsonOutput(result) return nil } var getRPCEndpointsCommand = &cli.Command{ Name: "getrpcendpoints", Usage: "gets GoCryptoTrader endpoints info", Action: getRPCEndpoints, } func getRPCEndpoints(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetRPCEndpoints(context.Background(), &gctrpc.GetRPCEndpointsRequest{}, ) if err != nil { return err } jsonOutput(result) return nil } var getCommunicationRelayersCommand = &cli.Command{ Name: "getcommsrelayers", Usage: "gets GoCryptoTrader communication relayers", Action: getCommunicationRelayers, } func getCommunicationRelayers(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetCommunicationRelayers(context.Background(), &gctrpc.GetCommunicationRelayersRequest{}, ) if err != nil { return err } jsonOutput(result) return nil } var getExchangesCommand = &cli.Command{ Name: "getexchanges", Usage: "gets a list of enabled or available exchanges", ArgsUsage: "", Action: getExchanges, Flags: []cli.Flag{ &cli.BoolFlag{ Name: "enabled", Usage: "whether to list enabled exchanges or not", }, }, } func getExchanges(c *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() var enabledOnly bool if c.IsSet("enabled") { enabledOnly = c.Bool("enabled") } client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetExchanges(context.Background(), &gctrpc.GetExchangesRequest{ Enabled: enabledOnly, }, ) if err != nil { return err } jsonOutput(result) return nil } var enableExchangeCommand = &cli.Command{ Name: "enableexchange", Usage: "enables an exchange", ArgsUsage: "", Action: enableExchange, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to enable", }, }, } func enableExchange(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "enableexchange") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.EnableExchange(context.Background(), &gctrpc.GenericExchangeNameRequest{ Exchange: exchangeName, }, ) if err != nil { return err } jsonOutput(result) return nil } var disableExchangeCommand = &cli.Command{ Name: "disableexchange", Usage: "disables an exchange", ArgsUsage: "", Action: disableExchange, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to disable", }, }, } func disableExchange(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "disableexchange") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.DisableExchange(context.Background(), &gctrpc.GenericExchangeNameRequest{ Exchange: exchangeName, }, ) if err != nil { return err } jsonOutput(result) return nil } var getExchangeOTPCommand = &cli.Command{ Name: "getexchangeotp", Usage: "gets a specific exchange OTP code", ArgsUsage: "", Action: getExchangeOTPCode, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the OTP code for", }, }, } func getExchangeOTPCode(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getexchangeotp") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetExchangeOTPCode(context.Background(), &gctrpc.GenericExchangeNameRequest{ Exchange: exchangeName, }, ) if err != nil { return err } jsonOutput(result) return nil } var getExchangeOTPsCommand = &cli.Command{ Name: "getexchangeotps", Usage: "gets all exchange OTP codes", Action: getExchangeOTPCodes, } func getExchangeOTPCodes(c *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetExchangeOTPCodes(context.Background(), &gctrpc.GetExchangeOTPsRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getExchangeInfoCommand = &cli.Command{ Name: "getexchangeinfo", Usage: "gets a specific exchanges info", ArgsUsage: "", Action: getExchangeInfo, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the info for", }, }, } func getExchangeInfo(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getexchangeinfo") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetExchangeInfo(context.Background(), &gctrpc.GenericExchangeNameRequest{ Exchange: exchangeName, }, ) if err != nil { return err } jsonOutput(result) return nil } var getTickerCommand = &cli.Command{ Name: "getticker", Usage: "gets the ticker for a specific currency pair and exchange", ArgsUsage: " ", Action: getTicker, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the ticker for", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair to get the ticker for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type of the currency pair to get the ticker for", }, }, } func getTicker(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getticker") } var exchangeName string var currencyPair string var assetType string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(1) } if !validPair(currencyPair) { return errInvalidPair } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(2) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetTicker(context.Background(), &gctrpc.GetTickerRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, AssetType: assetType, }, ) if err != nil { return err } jsonOutput(result) return nil } var getTickersCommand = &cli.Command{ Name: "gettickers", Usage: "gets all tickers for all enabled exchanges and currency pairs", Action: getTickers, } func getTickers(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetTickers(context.Background(), &gctrpc.GetTickersRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getOrderbookCommand = &cli.Command{ Name: "getorderbook", Usage: "gets the orderbook for a specific currency pair and exchange", ArgsUsage: " ", Action: getOrderbook, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the orderbook for", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair to get the orderbook for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type of the currency pair to get the orderbook for", }, }, } func getOrderbook(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getorderbook") } var exchangeName string var currencyPair string var assetType string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(1) } if !validPair(currencyPair) { return errInvalidPair } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(2) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetOrderbook(context.Background(), &gctrpc.GetOrderbookRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, AssetType: assetType, }, ) if err != nil { return err } jsonOutput(result) return nil } var getOrderbooksCommand = &cli.Command{ Name: "getorderbooks", Usage: "gets all orderbooks for all enabled exchanges and currency pairs", Action: getOrderbooks, } func getOrderbooks(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetOrderbooks(context.Background(), &gctrpc.GetOrderbooksRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getAccountInfoCommand = &cli.Command{ Name: "getaccountinfo", Usage: "gets the exchange account balance info", ArgsUsage: " ", Action: getAccountInfo, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the account info for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type to get the account info for", }, }, } func getAccountInfo(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getaccountinfo") } var exchange string if c.IsSet("exchange") { exchange = c.String("exchange") } else { exchange = c.Args().First() } var assetType string if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(1) } if !validExchange(exchange) { return errInvalidExchange } if !validAsset(assetType) { return errInvalidAsset } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetAccountInfo(context.Background(), &gctrpc.GetAccountInfoRequest{ Exchange: exchange, AssetType: assetType, }, ) if err != nil { return err } jsonOutput(result) return nil } var getAccountInfoStreamCommand = &cli.Command{ Name: "getaccountinfostream", Usage: "gets the account info stream for a specific exchange", ArgsUsage: " ", Action: getAccountInfoStream, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the account info stream from", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type to get the account info stream for", }, }, } func getAccountInfoStream(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getaccountinfostream") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } var assetType string if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(1) } if !validExchange(exchangeName) { return errInvalidExchange } if !validAsset(assetType) { return errInvalidAsset } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetAccountInfoStream(context.Background(), &gctrpc.GetAccountInfoRequest{Exchange: exchangeName, AssetType: assetType}) if err != nil { return err } for { resp, err := result.Recv() if err != nil { return err } err = clearScreen() if err != nil { return err } fmt.Printf("Account balance stream for %s:\n\n", exchangeName) fmt.Printf("%+v", resp) } } var updateAccountInfoCommand = &cli.Command{ Name: "updateaccountinfo", Usage: "updates the exchange account balance info", ArgsUsage: " ", Action: updateAccountInfo, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the account info for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type to get the account info for", }, }, } func updateAccountInfo(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "updateaccountinfo") } var exchange string if c.IsSet("exchange") { exchange = c.String("exchange") } else { exchange = c.Args().First() } var assetType string if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(1) } if !validExchange(exchange) { return errInvalidExchange } if !validAsset(assetType) { return errInvalidAsset } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.UpdateAccountInfo(context.Background(), &gctrpc.GetAccountInfoRequest{ Exchange: exchange, AssetType: assetType, }, ) if err != nil { return err } jsonOutput(result) return nil } var getConfigCommand = &cli.Command{ Name: "getconfig", Usage: "gets the config", Action: getConfig, } func getConfig(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetConfig(context.Background(), &gctrpc.GetConfigRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getPortfolioCommand = &cli.Command{ Name: "getportfolio", Usage: "gets the portfolio", Action: getPortfolio, } func getPortfolio(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetPortfolio(context.Background(), &gctrpc.GetPortfolioRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getPortfolioSummaryCommand = &cli.Command{ Name: "getportfoliosummary", Usage: "gets the portfolio summary", Action: getPortfolioSummary, } func getPortfolioSummary(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetPortfolioSummary(context.Background(), &gctrpc.GetPortfolioSummaryRequest{}) if err != nil { return err } jsonOutput(result) return nil } var addPortfolioAddressCommand = &cli.Command{ Name: "addportfolioaddress", Usage: "adds an address to the portfolio", ArgsUsage: "
", Action: addPortfolioAddress, Flags: []cli.Flag{ &cli.StringFlag{ Name: "address", Usage: "the address to add to the portfolio", }, &cli.StringFlag{ Name: "coin_type", Usage: "the coin type e.g ('BTC')", }, &cli.StringFlag{ Name: "description", Usage: "description of the address", }, &cli.Float64Flag{ Name: "balance", Usage: "balance of the address", }, &cli.BoolFlag{ Name: "cold_storage", Usage: "true/false if address is cold storage", }, &cli.StringFlag{ Name: "supported_exchanges", Usage: "common separated list of exchanges supported by this address for withdrawals", }, }, } func addPortfolioAddress(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "addportfolioaddress") } var address string var coinType string var description string var balance float64 var supportedExchanges string var coldstorage bool if c.IsSet("address") { address = c.String("address") } else { address = c.Args().First() } if c.IsSet("coin_type") { coinType = c.String("coin_type") } else { coinType = c.Args().Get(1) } if c.IsSet("description") { description = c.String("description") } else { description = c.Args().Get(2) } var err error if c.IsSet("balance") { balance = c.Float64("balance") } else if c.Args().Get(3) != "" { balance, err = strconv.ParseFloat(c.Args().Get(3), 64) if err != nil { return err } } if c.IsSet("cold_storage") { coldstorage = c.Bool("cold_storage") } else { tv, errBool := strconv.ParseBool(c.Args().Get(4)) if errBool == nil { coldstorage = tv } } if c.IsSet("supported_exchanges") { supportedExchanges = c.String("supported_exchanges") } else { supportedExchanges = c.Args().Get(5) } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.AddPortfolioAddress(context.Background(), &gctrpc.AddPortfolioAddressRequest{ Address: address, CoinType: coinType, Description: description, Balance: balance, SupportedExchanges: supportedExchanges, ColdStorage: coldstorage, }, ) if err != nil { return err } jsonOutput(result) return nil } var removePortfolioAddressCommand = &cli.Command{ Name: "removeportfolioaddress", Usage: "removes an address from the portfolio", ArgsUsage: "
", Action: removePortfolioAddress, Flags: []cli.Flag{ &cli.StringFlag{ Name: "address", Usage: "the address to add to the portfolio", }, &cli.StringFlag{ Name: "coin_type", Usage: "the coin type e.g ('BTC')", }, &cli.StringFlag{ Name: "description", Usage: "description of the address", }, }, } func removePortfolioAddress(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "removeportfolioaddress") } var address string var coinType string var description string if c.IsSet("address") { address = c.String("address") } else { address = c.Args().First() } if c.IsSet("coin_type") { coinType = c.String("coin_type") } else { coinType = c.Args().Get(1) } if c.IsSet("description") { description = c.String("description") } else { description = c.Args().Get(2) } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.RemovePortfolioAddress(context.Background(), &gctrpc.RemovePortfolioAddressRequest{ Address: address, CoinType: coinType, Description: description, }, ) if err != nil { return err } jsonOutput(result) return nil } var getForexProvidersCommand = &cli.Command{ Name: "getforexproviders", Usage: "gets the available forex providers", Action: getForexProviders, } func getForexProviders(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetForexProviders(context.Background(), &gctrpc.GetForexProvidersRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getForexRatesCommand = &cli.Command{ Name: "getforexrates", Usage: "gets forex rates", Action: getForexRates, } func getForexRates(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetForexRates(context.Background(), &gctrpc.GetForexRatesRequest{}) if err != nil { return err } jsonOutput(result) return nil } var getOrdersCommand = &cli.Command{ Name: "getorders", Usage: "gets the open orders", ArgsUsage: " ", Action: getOrders, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get orders for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type to get orders for", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair to get orders for", }, &cli.StringFlag{ Name: "start", Usage: "start date, optional. Will filter any results before this date", Value: time.Now().AddDate(0, -1, 0).Format(common.SimpleTimeFormat), Destination: &startTime, }, &cli.StringFlag{ Name: "end", Usage: "end date, optional. Will filter any results after this date", Value: time.Now().Format(common.SimpleTimeFormat), Destination: &endTime, }, }, } func getOrders(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getorders") } var exchangeName string var assetType string var currencyPair string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(1) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(2) } if !validPair(currencyPair) { return errInvalidPair } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } if !c.IsSet("start") { if c.Args().Get(3) != "" { startTime = c.Args().Get(3) } } if !c.IsSet("end") { if c.Args().Get(4) != "" { endTime = c.Args().Get(4) } } var s, e time.Time s, err = time.Parse(common.SimpleTimeFormat, startTime) if err != nil { return fmt.Errorf("invalid time format for start: %v", err) } e, err = time.Parse(common.SimpleTimeFormat, endTime) if err != nil { return fmt.Errorf("invalid time format for end: %v", err) } if e.Before(s) { return errors.New("start cannot be after end") } var conn *grpc.ClientConn conn, err = setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetOrders(context.Background(), &gctrpc.GetOrdersRequest{ Exchange: exchangeName, AssetType: assetType, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, StartDate: negateLocalOffset(s), EndDate: negateLocalOffset(e), }) if err != nil { return err } jsonOutput(result) return nil } var getOrderCommand = &cli.Command{ Name: "getorder", Usage: "gets the specified order info", ArgsUsage: " ", Action: getOrder, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the order for", }, &cli.StringFlag{ Name: "asset", Usage: "required asset type", }, &cli.StringFlag{ Name: "pair", Usage: "the pair to retrieve", }, &cli.StringFlag{ Name: "order_id", Usage: "the order id to retrieve", }, }, } func getOrder(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getorder") } var exchangeName string var orderID string var currencyPair string var assetType string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(1) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(2) } if !validPair(currencyPair) { return errInvalidPair } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } if c.IsSet("order_id") { orderID = c.String("order_id") } else { orderID = c.Args().Get(3) } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetOrder(context.Background(), &gctrpc.GetOrderRequest{ Exchange: exchangeName, OrderId: orderID, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, Asset: assetType, }) if err != nil { return err } jsonOutput(result) return nil } var submitOrderCommand = &cli.Command{ Name: "submitorder", Usage: "submit order submits an exchange order", ArgsUsage: " ", Action: submitOrder, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to submit the order for", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair", }, &cli.StringFlag{ Name: "side", Usage: "the order side to use (BUY OR SELL)", }, &cli.StringFlag{ Name: "type", Usage: "the order type (MARKET OR LIMIT)", }, &cli.Float64Flag{ Name: "amount", Usage: "the amount for the order", }, &cli.Float64Flag{ Name: "price", Usage: "the price for the order", }, &cli.StringFlag{ Name: "client_id", Usage: "the optional client order ID", }, &cli.StringFlag{ Name: "asset", Usage: "required asset type", }, }, } func submitOrder(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "submitorder") } var exchangeName string var currencyPair string var orderSide string var orderType string var amount float64 var price float64 var clientID string var assetType string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(1) } if !validPair(currencyPair) { return errInvalidPair } if c.IsSet("side") { orderSide = c.String("side") } else { orderSide = c.Args().Get(2) } if orderSide == "" { return errors.New("order side must be set") } if c.IsSet("type") { orderType = c.String("type") } else { orderType = c.Args().Get(3) } if orderType == "" { return errors.New("order type must be set") } if c.IsSet("amount") { amount = c.Float64("amount") } else if c.Args().Get(4) != "" { var err error amount, err = strconv.ParseFloat(c.Args().Get(4), 64) if err != nil { return err } } if amount == 0 { return errors.New("amount must be set") } // price is optional for market orders if c.IsSet("price") { price = c.Float64("price") } else if c.Args().Get(5) != "" { var err error price, err = strconv.ParseFloat(c.Args().Get(5), 64) if err != nil { return err } } if c.IsSet("client_id") { clientID = c.String("client_id") } else { clientID = c.Args().Get(6) } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(7) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.SubmitOrder(context.Background(), &gctrpc.SubmitOrderRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, Side: orderSide, OrderType: orderType, Amount: amount, Price: price, ClientId: clientID, AssetType: assetType, }) if err != nil { return err } jsonOutput(result) return nil } var simulateOrderCommand = &cli.Command{ Name: "simulateorder", Usage: "simulate order simulates an exchange order", ArgsUsage: " ", Action: simulateOrder, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to simulate the order for", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair", }, &cli.StringFlag{ Name: "side", Usage: "the order side to use (BUY OR SELL)", }, &cli.Float64Flag{ Name: "amount", Usage: "the amount for the order", }, }, } func simulateOrder(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "simulateorder") } var exchangeName string var currencyPair string var orderSide string var amount float64 if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(1) } if !validPair(currencyPair) { return errInvalidPair } if c.IsSet("side") { orderSide = c.String("side") } else { orderSide = c.Args().Get(2) } if orderSide == "" { return errors.New("side must be set") } if c.IsSet("amount") { amount = c.Float64("amount") } else if c.Args().Get(3) != "" { var err error amount, err = strconv.ParseFloat(c.Args().Get(3), 64) if err != nil { return err } } if amount == 0 { return errors.New("amount must be set") } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.SimulateOrder(context.Background(), &gctrpc.SimulateOrderRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, Side: orderSide, Amount: amount, }) if err != nil { return err } jsonOutput(result) return nil } var whaleBombCommand = &cli.Command{ Name: "whalebomb", Usage: "whale bomb finds the amount required to reach a price target", ArgsUsage: " ", Action: whaleBomb, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to whale bomb", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair", }, &cli.StringFlag{ Name: "side", Usage: "the order side to use (BUY OR SELL)", }, &cli.Float64Flag{ Name: "price", Usage: "the price target", }, }, } func whaleBomb(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "whalebomb") } var exchangeName string var currencyPair string var orderSide string var price float64 if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(1) } if !validPair(currencyPair) { return errInvalidPair } if c.IsSet("side") { orderSide = c.String("side") } else { orderSide = c.Args().Get(2) } if orderSide == "" { return errors.New("order side must be set") } if c.IsSet("price") { price = c.Float64("price") } else if c.Args().Get(3) != "" { var err error price, err = strconv.ParseFloat(c.Args().Get(3), 64) if err != nil { return err } } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.WhaleBomb(context.Background(), &gctrpc.WhaleBombRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, Side: orderSide, PriceTarget: price, }) if err != nil { return err } jsonOutput(result) return nil } var cancelOrderCommand = &cli.Command{ Name: "cancelorder", Usage: "cancel order cancels an exchange order", ArgsUsage: " ", Action: cancelOrder, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to cancel the order for", }, &cli.StringFlag{ Name: "account_id", Usage: "the account id", }, &cli.StringFlag{ Name: "order_id", Usage: "the order id", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair to cancel the order for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type", }, &cli.Float64Flag{ Name: "wallet_address", Usage: "the wallet address", }, &cli.StringFlag{ Name: "side", Usage: "the order side", }, }, } func cancelOrder(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "cancelorder") } var exchangeName string var accountID string var orderID string var currencyPair string var assetType string var walletAddress string var orderSide string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("account_id") { accountID = c.String("account_id") } else { accountID = c.Args().Get(1) } if c.IsSet("order_id") { orderID = c.String("order_id") } else { orderID = c.Args().Get(2) } if orderID == "" { return errors.New("an order ID must be set") } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(3) } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(4) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } if c.IsSet("wallet_address") { walletAddress = c.String("wallet_address") } else { walletAddress = c.Args().Get(5) } if c.IsSet("side") { orderSide = c.String("side") } else { orderSide = c.Args().Get(6) } // pair is optional, but if it's set, do a validity check var p currency.Pair if len(currencyPair) > 0 { if !validPair(currencyPair) { return errInvalidPair } var err error p, err = currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.CancelOrder(context.Background(), &gctrpc.CancelOrderRequest{ Exchange: exchangeName, AccountId: accountID, OrderId: orderID, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, AssetType: assetType, WalletAddress: walletAddress, Side: orderSide, }) if err != nil { return err } jsonOutput(result) return nil } var cancelBatchOrdersCommand = &cli.Command{ Name: "cancelbatchorders", Usage: "cancel batch orders cancels a list of exchange orders (comma separated)", ArgsUsage: " ", Action: cancelBatchOrders, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to cancel the order for", }, &cli.StringFlag{ Name: "account_id", Usage: "the account id", }, &cli.StringFlag{ Name: "order_ids", Usage: "the comma separated orders id-s", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair to cancel the order for", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type", }, &cli.Float64Flag{ Name: "wallet_address", Usage: "the wallet address", }, &cli.StringFlag{ Name: "side", Usage: "the order side", }, }, } func cancelBatchOrders(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "cancelbatchorders") } var exchangeName string var accountID string var orderID string var currencyPair string var assetType string var walletAddress string var orderSide string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("account_id") { accountID = c.String("account_id") } else { accountID = c.Args().Get(1) } if c.IsSet("order_ids") { orderID = c.String("order_ids") } else { orderID = c.Args().Get(2) } if orderID == "" { return errors.New("an order ID must be set") } if c.IsSet("pair") { currencyPair = c.String("pair") } else { currencyPair = c.Args().Get(3) } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(4) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } if c.IsSet("wallet_address") { walletAddress = c.String("wallet_address") } else { walletAddress = c.Args().Get(5) } if c.IsSet("side") { orderSide = c.String("side") } else { orderSide = c.Args().Get(6) } // pair is optional, but if it's set, do a validity check var p currency.Pair if len(currencyPair) > 0 { if !validPair(currencyPair) { return errInvalidPair } var err error p, err = currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.CancelBatchOrders(context.Background(), &gctrpc.CancelBatchOrdersRequest{ Exchange: exchangeName, AccountId: accountID, OrdersId: orderID, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, AssetType: assetType, WalletAddress: walletAddress, Side: orderSide, }) if err != nil { return err } jsonOutput(result) return nil } var cancelAllOrdersCommand = &cli.Command{ Name: "cancelallorders", Usage: "cancels all orders (all or by exchange name)", ArgsUsage: "", Action: cancelAllOrders, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to cancel all orders on", }, }, } func cancelAllOrders(c *cli.Context) error { var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } // exchange name is an optional param if exchangeName != "" { if !validExchange(exchangeName) { return errInvalidExchange } } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.CancelAllOrders(context.Background(), &gctrpc.CancelAllOrdersRequest{ Exchange: exchangeName, }) if err != nil { return err } jsonOutput(result) return nil } var getEventsCommand = &cli.Command{ Name: "getevents", Usage: "gets all events", Action: getEvents, } func getEvents(_ *cli.Context) error { conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetEvents(context.Background(), &gctrpc.GetEventsRequest{}) if err != nil { return err } jsonOutput(result) return nil } var addEventCommand = &cli.Command{ Name: "addevent", Usage: "adds an event", ArgsUsage: " ", Action: addEvent, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to add an event for", }, &cli.StringFlag{ Name: "item", Usage: "the item to trigger the event", }, &cli.StringFlag{ Name: "condition", Usage: "the condition for the event", }, &cli.Float64Flag{ Name: "price", Usage: "the price to trigger the event", }, &cli.BoolFlag{ Name: "check_bids", Usage: "whether to check the bids", }, &cli.BoolFlag{ Name: "check_asks", Usage: "whether to check the asks", }, &cli.Float64Flag{ Name: "orderbook_amount", Usage: "the orderbook amount to trigger the event", }, &cli.StringFlag{ Name: "pair", Usage: "the currency pair", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type", }, &cli.StringFlag{ Name: "action", Usage: "the action for the event to perform upon trigger", }, }, } func addEvent(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "addevent") } var exchangeName string var item string var condition string var price float64 var checkBids bool var checkAsks bool var orderbookAmount float64 var currencyPair string var assetType string var action string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { return fmt.Errorf("exchange name is required") } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("item") { item = c.String("item") } else { return fmt.Errorf("item is required") } if c.IsSet("condition") { condition = c.String("condition") } else { return fmt.Errorf("condition is required") } if c.IsSet("price") { price = c.Float64("price") } if c.IsSet("check_bids") { checkBids = c.Bool("check_bids") } if c.IsSet("check_asks") { checkAsks = c.Bool("check_asks") } if c.IsSet("orderbook_amount") { orderbookAmount = c.Float64("orderbook_amount") } if c.IsSet("pair") { currencyPair = c.String("pair") } else { return fmt.Errorf("currency pair is required") } if !validPair(currencyPair) { return errInvalidPair } if c.IsSet("asset") { assetType = c.String("asset") } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } if c.IsSet("action") { action = c.String("action") } else { return fmt.Errorf("action is required") } p, err := currency.NewPairDelimiter(currencyPair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.AddEvent(context.Background(), &gctrpc.AddEventRequest{ Exchange: exchangeName, Item: item, ConditionParams: &gctrpc.ConditionParams{ Condition: condition, Price: price, CheckBids: checkBids, CheckAsks: checkAsks, OrderbookAmount: orderbookAmount, }, Pair: &gctrpc.CurrencyPair{ Delimiter: p.Delimiter, Base: p.Base.String(), Quote: p.Quote.String(), }, AssetType: assetType, Action: action, }) if err != nil { return err } jsonOutput(result) return nil } var removeEventCommand = &cli.Command{ Name: "removeevent", Usage: "removes an event", ArgsUsage: "", Action: removeEvent, Flags: []cli.Flag{ &cli.StringFlag{ Name: "event_id", Usage: "the event id to remove", }, }, } func removeEvent(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "removeevent") } var eventID int64 if c.IsSet("event_id") { eventID = c.Int64("event_id") } else if c.Args().Get(0) != "" { var err error eventID, err = strconv.ParseInt(c.Args().Get(0), 10, 64) if err != nil { return err } } if eventID == 0 { return errors.New("event id must be specified") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.RemoveEvent(context.Background(), &gctrpc.RemoveEventRequest{Id: eventID}) if err != nil { return err } jsonOutput(result) return nil } var getCryptocurrencyDepositAddressesCommand = &cli.Command{ Name: "getcryptocurrencydepositaddresses", Usage: "gets the cryptocurrency deposit addresses for an exchange", ArgsUsage: "", Action: getCryptocurrencyDepositAddresses, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the cryptocurrency deposit addresses for", }, }, } func getCryptocurrencyDepositAddresses(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getcryptocurrencydepositaddresses") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetCryptocurrencyDepositAddresses(context.Background(), &gctrpc.GetCryptocurrencyDepositAddressesRequest{Exchange: exchangeName}) if err != nil { return err } jsonOutput(result) return nil } var getCryptocurrencyDepositAddressCommand = &cli.Command{ Name: "getcryptocurrencydepositaddress", Usage: "gets the cryptocurrency deposit address for an exchange and cryptocurrency", ArgsUsage: " ", Action: getCryptocurrencyDepositAddress, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the cryptocurrency deposit address for", }, &cli.StringFlag{ Name: "cryptocurrency", Usage: "the cryptocurrency to get the deposit address for", }, }, } func getCryptocurrencyDepositAddress(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getcryptocurrencydepositaddresses") } var exchangeName string var cryptocurrency string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("cryptocurrency") { cryptocurrency = c.String("cryptocurrency") } else if c.Args().Get(1) != "" { cryptocurrency = c.Args().Get(1) } if cryptocurrency == "" { return errors.New("cryptocurrency must be set") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetCryptocurrencyDepositAddress(context.Background(), &gctrpc.GetCryptocurrencyDepositAddressRequest{ Exchange: exchangeName, Cryptocurrency: cryptocurrency, }, ) if err != nil { return err } jsonOutput(result) return nil } var withdrawCryptocurrencyFundsCommand = &cli.Command{ Name: "withdrawcryptofunds", Usage: "withdraws cryptocurrency funds from the desired exchange", ArgsUsage: "
", Action: withdrawCryptocurrencyFunds, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to withdraw from", }, &cli.StringFlag{ Name: "currency", Usage: "the cryptocurrency to withdraw funds from", }, &cli.StringFlag{ Name: "address", Usage: "address to withdraw to", }, &cli.StringFlag{ Name: "addresstag", Usage: "address tag/memo", }, &cli.Float64Flag{ Name: "amount", Usage: "amount of funds to withdraw", }, &cli.Float64Flag{ Name: "fee", Usage: "fee to submit with request", }, &cli.StringFlag{ Name: "description", Usage: "description to submit with request", }, }, } func withdrawCryptocurrencyFunds(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "withdrawcryptofunds") } var exchange, cur, address, addressTag, description string var amount, fee float64 if c.IsSet("exchange") { exchange = c.String("exchange") } else if c.Args().Get(0) != "" { exchange = c.Args().Get(0) } if !validExchange(exchange) { return errInvalidExchange } if c.IsSet("currency") { cur = c.String("currency") } else if c.Args().Get(1) != "" { cur = c.Args().Get(1) } if c.IsSet("amount") { amount = c.Float64("amount") } else if c.Args().Get(2) != "" { amountStr, err := strconv.ParseFloat(c.Args().Get(2), 64) if err == nil { amount = amountStr } } if c.IsSet("address") { address = c.String("address") } else if c.Args().Get(3) != "" { address = c.Args().Get(3) } if c.IsSet("addresstag") { addressTag = c.String("addresstag") } else if c.Args().Get(4) != "" { addressTag = c.Args().Get(4) } if c.IsSet("fee") { fee = c.Float64("fee") } else if c.Args().Get(5) != "" { feeStr, err := strconv.ParseFloat(c.Args().Get(5), 64) if err == nil { fee = feeStr } } if c.IsSet("description") { description = c.String("description") } else if c.Args().Get(6) != "" { description = c.Args().Get(6) } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.WithdrawCryptocurrencyFunds(context.Background(), &gctrpc.WithdrawCryptoRequest{ Exchange: exchange, Currency: cur, Address: address, AddressTag: addressTag, Amount: amount, Fee: fee, Description: description, }, ) if err != nil { return err } jsonOutput(result) return nil } var withdrawFiatFundsCommand = &cli.Command{ Name: "withdrawfiatfunds", Usage: "withdraws fiat funds from the desired exchange", ArgsUsage: " ", Action: withdrawFiatFunds, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to withdraw from", }, &cli.StringFlag{ Name: "currency", Usage: "the fiat currency to withdraw funds from", }, &cli.Float64Flag{ Name: "amount", Usage: "amount of funds to withdraw", }, &cli.StringFlag{ Name: "bankaccountid", Usage: "ID of bank account to use", }, &cli.StringFlag{ Name: "description", Usage: "description to submit with request", }, }, } func withdrawFiatFunds(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "withdrawfiatfunds") } var exchange, cur, description, bankAccountID string var amount float64 if c.IsSet("exchange") { exchange = c.String("exchange") } else if c.Args().Get(0) != "" { exchange = c.Args().Get(0) } if !validExchange(exchange) { return errInvalidExchange } if c.IsSet("currency") { cur = c.String("currency") } else if c.Args().Get(1) != "" { cur = c.Args().Get(1) } if c.IsSet("amount") { amount = c.Float64("amount") } else if c.Args().Get(2) != "" { amountStr, err := strconv.ParseFloat(c.Args().Get(2), 64) if err == nil { amount = amountStr } } if c.IsSet("bankaccountid") { bankAccountID = c.String("bankaccountid") } else if c.Args().Get(3) != "" { bankAccountID = c.Args().Get(3) } if c.IsSet("description") { description = c.String("description") } else if c.Args().Get(4) != "" { description = c.Args().Get(4) } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.WithdrawFiatFunds(context.Background(), &gctrpc.WithdrawFiatRequest{ Exchange: exchange, Currency: cur, Amount: amount, Description: description, BankAccountId: bankAccountID, }, ) if err != nil { return err } jsonOutput(result) return nil } var withdrawalRequestCommand = &cli.Command{ Name: "withdrawalrequesthistory", Usage: "retrieve previous withdrawal request details", ArgsUsage: " ", Subcommands: []*cli.Command{ { Name: "byid", Usage: "id", ArgsUsage: "", Flags: []cli.Flag{ &cli.StringFlag{ Name: "id", Usage: "withdrawal id", }, }, Action: withdrawlRequestByID, }, { Name: "byexchangeid", Usage: "exchange id", ArgsUsage: "", Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "exchange name", }, &cli.StringFlag{ Name: "id", Usage: "withdrawal id", }, }, Action: withdrawlRequestByExchangeID, }, { Name: "byexchange", Usage: "exchange limit", ArgsUsage: "", Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "exchange name", }, &cli.Int64Flag{ Name: "limit", Usage: "max number of withdrawals to return", }, &cli.StringFlag{ Name: "currency", Usage: "", }, }, Action: withdrawlRequestByExchangeID, }, { Name: "bydate", Usage: "exchange start end limit", ArgsUsage: " ", Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the currency used in to withdraw", }, &cli.StringFlag{ Name: "start", Usage: "the start date to get withdrawals from. Any withdrawal before this date will be filtered", Value: time.Now().AddDate(0, -1, 0).Format(common.SimpleTimeFormat), Destination: &startTime, }, &cli.StringFlag{ Name: "end", Usage: "the end date to get withdrawals from. Any withdrawal after this date will be filtered", Value: time.Now().Format(common.SimpleTimeFormat), Destination: &endTime, }, &cli.Int64Flag{ Name: "limit", Usage: "max number of withdrawals to return", }, }, Action: withdrawlRequestByDate, }, }, } func withdrawlRequestByID(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { cli.ShowSubcommandHelp(c) return nil } var ID string if c.IsSet("id") { ID = c.String("id") } else { ID = c.Args().First() } if ID == "" { return errors.New("an ID must be specified") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.WithdrawalEventByID(context.Background(), &gctrpc.WithdrawalEventByIDRequest{ Id: ID, }, ) if err != nil { return err } jsonOutput(result) return nil } func withdrawlRequestByExchangeID(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { cli.ShowSubcommandHelp(c) return nil } var exchange, currency string if c.IsSet("exchange") { exchange = c.String("exchange") } else { exchange = c.Args().First() } var limit, limitStr int64 var ID string var err error if c.Command.Name == "byexchangeid" { if c.IsSet("id") { ID = c.String("id") } else { ID = c.Args().Get(1) } if ID == "" { return errors.New("an ID must be specified") } limit = 1 } else { if c.IsSet("limit") { limit = c.Int64("limit") } else if c.Args().Get(1) != "" { limitStr, err = strconv.ParseInt(c.Args().Get(1), 10, 64) if err != nil { return err } if limitStr > math.MaxInt32 { return fmt.Errorf("limit greater than max size: %v", math.MaxInt32) } limit = limitStr } if c.IsSet("currency") { currency = c.String("currency") } } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.WithdrawalEventsByExchange(context.Background(), &gctrpc.WithdrawalEventsByExchangeRequest{ Exchange: exchange, Id: ID, Limit: int32(limit), Currency: currency, }, ) if err != nil { return err } jsonOutput(result) return nil } func withdrawlRequestByDate(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { cli.ShowSubcommandHelp(c) return nil } var exchange string var limit, limitStr int64 var err error if c.IsSet("exchange") { exchange = c.String("exchange") } else { exchange = c.Args().First() } if !c.IsSet("start") { if c.Args().Get(1) != "" { startTime = c.Args().Get(1) } } if !c.IsSet("end") { if c.Args().Get(2) != "" { endTime = c.Args().Get(2) } } if c.IsSet("limit") { limit = c.Int64("limit") } else if c.Args().Get(3) != "" { limitStr, err = strconv.ParseInt(c.Args().Get(3), 10, 64) if err != nil { return err } if limitStr > math.MaxInt32 { return fmt.Errorf("limit greater than max size: %v", math.MaxInt32) } limit = limitStr } s, err := time.Parse(common.SimpleTimeFormat, startTime) if err != nil { return fmt.Errorf("invalid time format for start: %v", err) } e, err := time.Parse(common.SimpleTimeFormat, endTime) if err != nil { return fmt.Errorf("invalid time format for end: %v", err) } if e.Before(s) { return errors.New("start cannot be after end") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.WithdrawalEventsByDate(context.Background(), &gctrpc.WithdrawalEventsByDateRequest{ Exchange: exchange, Start: negateLocalOffset(s), End: negateLocalOffset(e), Limit: int32(limit), }, ) if err != nil { return err } jsonOutput(result) return nil } var getLoggerDetailsCommand = &cli.Command{ Name: "getloggerdetails", Usage: "gets an individual loggers details", ArgsUsage: "", Action: getLoggerDetails, Flags: []cli.Flag{ &cli.StringFlag{ Name: "logger", Usage: "logger to get level details of", }, }, } func getLoggerDetails(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getloggerdetails") } var logger string if c.IsSet("logger") { logger = c.String("logger") } else { logger = c.Args().First() } if logger == "" { return errors.New("a logger must be specified") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetLoggerDetails(context.Background(), &gctrpc.GetLoggerDetailsRequest{ Logger: logger, }, ) if err != nil { return err } jsonOutput(result) return nil } var setLoggerDetailsCommand = &cli.Command{ Name: "setloggerdetails", Usage: "sets an individual loggers details", ArgsUsage: " ", Action: setLoggerDetails, Flags: []cli.Flag{ &cli.StringFlag{ Name: "logger", Usage: "logger to get level details of", }, &cli.StringFlag{ Name: "flags", Usage: "pipe separated value of loggers e.g INFO|WARN", }, }, } func setLoggerDetails(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "setloggerdetails") } var logger string var level string if c.IsSet("logger") { logger = c.String("logger") } else { logger = c.Args().First() } if logger == "" { return errors.New("a logger must be specified") } if c.IsSet("level") { level = c.String("level") } else { level = c.Args().Get(1) } if level == "" { return errors.New("level must be specified") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.SetLoggerDetails(context.Background(), &gctrpc.SetLoggerDetailsRequest{ Logger: logger, Level: level, }, ) if err != nil { return err } jsonOutput(result) return nil } var getOrderbookStreamCommand = &cli.Command{ Name: "getorderbookstream", Usage: "gets the orderbook stream for a specific currency pair and exchange", ArgsUsage: " ", Action: getOrderbookStream, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the orderbook from", }, &cli.StringFlag{ Name: "pair", Usage: "currency pair", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type of the currency pair", }, }, } func getOrderbookStream(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getorderbookstream") } var exchangeName string var pair string var assetType string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { pair = c.String("pair") } else { pair = c.Args().Get(1) } if !validPair(pair) { return errInvalidPair } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(2) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } p, err := currency.NewPairDelimiter(pair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetOrderbookStream(context.Background(), &gctrpc.GetOrderbookStreamRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Base: p.Base.String(), Quote: p.Quote.String(), Delimiter: p.Delimiter, }, AssetType: assetType, }, ) if err != nil { return err } for { resp, err := result.Recv() if err != nil { return err } err = clearScreen() if err != nil { return err } fmt.Printf("Orderbook stream for %s %s:\n\n", exchangeName, resp.Pair.String()) fmt.Println("\t\tBids\t\t\t\tAsks") fmt.Println() bidLen := len(resp.Bids) - 1 askLen := len(resp.Asks) - 1 var maxLen int if bidLen >= askLen { maxLen = bidLen } else { maxLen = askLen } for i := 0; i < maxLen; i++ { var bidAmount, bidPrice float64 if i <= bidLen { bidAmount = resp.Bids[i].Amount bidPrice = resp.Bids[i].Price } var askAmount, askPrice float64 if i <= askLen { askAmount = resp.Asks[i].Amount askPrice = resp.Asks[i].Price } fmt.Printf("%.8f %s @ %.8f %s\t\t%.8f %s @ %.8f %s\n", bidAmount, resp.Pair.Base, bidPrice, resp.Pair.Quote, askAmount, resp.Pair.Base, askPrice, resp.Pair.Quote) if i >= 49 { // limits orderbook display output break } } } } var getExchangeOrderbookStreamCommand = &cli.Command{ Name: "getexchangeorderbookstream", Usage: "gets a stream for all orderbooks associated with an exchange", ArgsUsage: "", Action: getExchangeOrderbookStream, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the orderbook from", }, }, } func getExchangeOrderbookStream(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getexchangeorderbookstream") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetExchangeOrderbookStream(context.Background(), &gctrpc.GetExchangeOrderbookStreamRequest{ Exchange: exchangeName, }) if err != nil { return err } for { resp, err := result.Recv() if err != nil { return err } err = clearScreen() if err != nil { return err } fmt.Printf("Orderbook streamed for %s %s", exchangeName, resp.Pair.String()) } } var getTickerStreamCommand = &cli.Command{ Name: "gettickerstream", Usage: "gets the ticker stream for a specific currency pair and exchange", ArgsUsage: " ", Action: getTickerStream, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the ticker from", }, &cli.StringFlag{ Name: "pair", Usage: "currency pair", }, &cli.StringFlag{ Name: "asset", Usage: "the asset type of the currency pair", }, }, } func getTickerStream(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "gettickerstream") } var exchangeName string var pair string var assetType string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } if c.IsSet("pair") { pair = c.String("pair") } else { pair = c.Args().Get(1) } if !validPair(pair) { return errInvalidPair } if c.IsSet("asset") { assetType = c.String("asset") } else { assetType = c.Args().Get(2) } assetType = strings.ToLower(assetType) if !validAsset(assetType) { return errInvalidAsset } p, err := currency.NewPairDelimiter(pair, pairDelimiter) if err != nil { return err } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetTickerStream(context.Background(), &gctrpc.GetTickerStreamRequest{ Exchange: exchangeName, Pair: &gctrpc.CurrencyPair{ Base: p.Base.String(), Quote: p.Quote.String(), Delimiter: p.Delimiter, }, AssetType: assetType, }, ) if err != nil { return err } for { resp, err := result.Recv() if err != nil { return err } err = clearScreen() if err != nil { return err } fmt.Printf("Ticker stream for %s %s:\n", exchangeName, resp.Pair.String()) fmt.Println() fmt.Printf("LAST: %f\n HIGH: %f\n LOW: %f\n BID: %f\n ASK: %f\n VOLUME: %f\n PRICEATH: %f\n LASTUPDATED: %d\n", resp.Last, resp.High, resp.Low, resp.Bid, resp.Ask, resp.Volume, resp.PriceAth, resp.LastUpdated) } } var getExchangeTickerStreamCommand = &cli.Command{ Name: "getexchangetickerstream", Usage: "gets a stream for all tickers associated with an exchange", ArgsUsage: "", Action: getExchangeTickerStream, Flags: []cli.Flag{ &cli.StringFlag{ Name: "exchange", Usage: "the exchange to get the ticker from", }, }, } func getExchangeTickerStream(c *cli.Context) error { if c.NArg() == 0 && c.NumFlags() == 0 { return cli.ShowCommandHelp(c, "getexchangetickerstream") } var exchangeName string if c.IsSet("exchange") { exchangeName = c.String("exchange") } else { exchangeName = c.Args().First() } if !validExchange(exchangeName) { return errInvalidExchange } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetExchangeTickerStream(context.Background(), &gctrpc.GetExchangeTickerStreamRequest{ Exchange: exchangeName, }) if err != nil { return err } for { resp, err := result.Recv() if err != nil { return err } fmt.Printf("Ticker stream for %s %s:\n", exchangeName, resp.Pair.String()) fmt.Printf("LAST: %f HIGH: %f LOW: %f BID: %f ASK: %f VOLUME: %f PRICEATH: %f LASTUPDATED: %d\n", resp.Last, resp.High, resp.Low, resp.Bid, resp.Ask, resp.Volume, resp.PriceAth, resp.LastUpdated) } } var getAuditEventCommand = &cli.Command{ Name: "getauditevent", Usage: "gets audit events matching query parameters", ArgsUsage: " ", Action: getAuditEvent, Flags: []cli.Flag{ &cli.StringFlag{ Name: "start", Aliases: []string{"s"}, Usage: "start date to search", Value: time.Now().Add(-time.Hour).Format(common.SimpleTimeFormat), Destination: &startTime, }, &cli.StringFlag{ Name: "end", Aliases: []string{"e"}, Usage: "end time to search", Value: time.Now().Format(common.SimpleTimeFormat), Destination: &endTime, }, &cli.StringFlag{ Name: "order", Aliases: []string{"o"}, Usage: "order results by ascending/descending", Value: "asc", Destination: &order, }, &cli.IntFlag{ Name: "limit", Aliases: []string{"l"}, Usage: "how many results to retrieve", Value: 100, Destination: &limit, }, }, } func getAuditEvent(c *cli.Context) error { if !c.IsSet("start") { if c.Args().Get(0) != "" { startTime = c.Args().Get(0) } } if !c.IsSet("end") { if c.Args().Get(1) != "" { endTime = c.Args().Get(1) } } if !c.IsSet("order") { if c.Args().Get(2) != "" { order = c.Args().Get(2) } } if !c.IsSet("limit") { if c.Args().Get(3) != "" { limitStr, err := strconv.ParseInt(c.Args().Get(3), 10, 64) if err == nil { limit = int(limitStr) } } } s, err := time.Parse(common.SimpleTimeFormat, startTime) if err != nil { return fmt.Errorf("invalid time format for start: %v", err) } e, err := time.Parse(common.SimpleTimeFormat, endTime) if err != nil { return fmt.Errorf("invalid time format for end: %v", err) } if e.Before(s) { return errors.New("start cannot be after end") } conn, err := setupClient() if err != nil { return err } defer conn.Close() client := gctrpc.NewGoCryptoTraderClient(conn) result, err := client.GetAuditEvent(context.Background(), &gctrpc.GetAuditEventRequest{ StartDate: negateLocalOffset(s), EndDate: negateLocalOffset(e), Limit: int32(limit), OrderBy: order, }) if err != nil { return err } jsonOutput(result) return nil } var uuid, filename, path string var gctScriptCommand = &cli.Command{ Name: "script", Usage: "execute scripting management command", ArgsUsage: " ", Subcommands: []*cli.Command{ { Name: "execute", Usage: "execute script filename", ArgsUsage: " ", Flags: []cli.Flag{ &cli.StringFlag{ Name: "filename", Usage: "the script filename", Destination: &filename, }, &cli.StringFlag{ Name: "path", Usage: "the directory of the script file", Destination: &path, }, }, Action: gctScriptExecute, }, { Name: "query", Usage: "query running virtual machine", Flags: []cli.Flag{ &cli.StringFlag{ Name: "uuid", Usage: "the unique id of the script in memory", Destination: &uuid, }, }, Action: gctScriptQuery, }, { Name: "read", Usage: "read script", Flags: []cli.Flag{ &cli.StringFlag{ Name: "name", Usage: "the script name", Destination: &uuid, }, }, Action: gctScriptRead, }, { Name: "status", Usage: "get status of running scripts", Action: gctScriptStatus, }, { Name: "list", Usage: "lists all scripts in default scriptpath", Action: gctScriptList, }, { Name: "stop", Usage: "terminate running script", Flags: []cli.Flag{ &cli.StringFlag{ Name: "uuid", Usage: "the unique id of the script in memory", Destination: &uuid, }, }, Action: gctScriptStop, }, { Name: "stopall", Usage: "terminate running script", Action: gctScriptStopAll, }, { Name: "upload", Usage: "upload a new script/archive", Flags: []cli.Flag{ &cli.StringFlag{ Name: "path", Usage: " to single script or zip collection", Destination: &filename, }, &cli.BoolFlag{ Name: "overwrite", Usage: "", }, &cli.BoolFlag{ Name: "archived", Usage: "", }, }, Action: gctScriptUpload, }, { Name: "autoload", Usage: "add or remove script from autoload list", Flags: []cli.Flag{ &cli.StringFlag{ Name: "command", Usage: "", }, &cli.StringFlag{ Name: "script", Usage: "