add examples for placing orders with ws updates. Add some response types.
This commit is contained in:
@@ -1,7 +1,7 @@
|
|||||||
# bitget-api
|
# bitget-api
|
||||||
[](https://github.com/tiagosiebler/bitget-api/actions/workflows/integrationtest.yml) [][1] [][1] [][1]
|
[](https://github.com/tiagosiebler/bitget-api/actions/workflows/integrationtest.yml) [][1] [][1] [][1]
|
||||||
[][1]
|
[][1]
|
||||||
[](https://www.codefactor.io/repository/github/tiagosiebler/bitget-api)
|
[](https://www.codefactor.io/repository/github/tiagosiebler/bitget-api) [](https://t.me/nodetraders)
|
||||||
|
|
||||||
|
|
||||||
[][1]
|
[][1]
|
||||||
|
|||||||
104
examples/rest-trade-futures.ts
Normal file
104
examples/rest-trade-futures.ts
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
import { FuturesClient, WebsocketClient } from '../src/index';
|
||||||
|
|
||||||
|
// or
|
||||||
|
// import { SpotClient } from 'bitget-api';
|
||||||
|
|
||||||
|
// read from environmental variables
|
||||||
|
const API_KEY = process.env.API_KEY_COM;
|
||||||
|
const API_SECRET = process.env.API_SECRET_COM;
|
||||||
|
const API_PASS = process.env.API_PASS_COM;
|
||||||
|
|
||||||
|
const client = new FuturesClient({
|
||||||
|
apiKey: API_KEY,
|
||||||
|
// apiKey: 'apiKeyHere',
|
||||||
|
apiSecret: API_SECRET,
|
||||||
|
// apiSecret: 'apiSecretHere',
|
||||||
|
apiPass: API_PASS,
|
||||||
|
// apiPass: 'apiPassHere',
|
||||||
|
});
|
||||||
|
|
||||||
|
const wsClient = new WebsocketClient({
|
||||||
|
apiKey: API_KEY,
|
||||||
|
apiSecret: API_SECRET,
|
||||||
|
apiPass: API_PASS,
|
||||||
|
});
|
||||||
|
|
||||||
|
function logWSEvent(type, data) {
|
||||||
|
console.log(new Date(), `WS ${type} event: `, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// simple sleep function
|
||||||
|
function promiseSleep(milliseconds) {
|
||||||
|
return new Promise((resolve) => setTimeout(resolve, milliseconds));
|
||||||
|
}
|
||||||
|
|
||||||
|
// WARNING: for sensitive math you should be using a library such as decimal.js!
|
||||||
|
function roundDown(value, decimals) {
|
||||||
|
return Number(
|
||||||
|
Math.floor(parseFloat(value + 'e' + decimals)) + 'e-' + decimals
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** This is a simple script wrapped in a immediately invoked function expression, designed to check for any available BTC balance and immediately sell the full amount for USDT */
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
// Add event listeners to log websocket events on account
|
||||||
|
wsClient.on('update', (data) => logWSEvent('update', data));
|
||||||
|
wsClient.on('open', (data) => logWSEvent('open', data));
|
||||||
|
wsClient.on('response', (data) => logWSEvent('response', data));
|
||||||
|
wsClient.on('reconnect', (data) => logWSEvent('reconnect', data));
|
||||||
|
wsClient.on('reconnected', (data) => logWSEvent('reconnected', data));
|
||||||
|
wsClient.on('authenticated', (data) => logWSEvent('authenticated', data));
|
||||||
|
wsClient.on('exception', (data) => logWSEvent('exception', data));
|
||||||
|
|
||||||
|
// Subscribe to private account topics
|
||||||
|
wsClient.subscribeTopic('UMCBL', 'account');
|
||||||
|
// : position updates
|
||||||
|
wsClient.subscribeTopic('UMCBL', 'positions');
|
||||||
|
// : order updates
|
||||||
|
wsClient.subscribeTopic('UMCBL', 'orders');
|
||||||
|
|
||||||
|
// wait briefly for ws to be ready (could also use the response or authenticated events, to make sure topics are subscribed to before starting)
|
||||||
|
await promiseSleep(2.5 * 1000);
|
||||||
|
|
||||||
|
const symbol = 'BTCUSDT_UMCBL';
|
||||||
|
const marginCoin = 'USDT';
|
||||||
|
|
||||||
|
const balanceResult = await client.getAccount(symbol, marginCoin);
|
||||||
|
const accountBalance = balanceResult.data;
|
||||||
|
// const balances = allBalances.filter((bal) => Number(bal.available) != 0);
|
||||||
|
const usdtAmount = accountBalance.available;
|
||||||
|
console.log('USDT balance: ', usdtAmount);
|
||||||
|
|
||||||
|
if (!usdtAmount) {
|
||||||
|
console.error('No USDT to trade');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const symbolRulesResult = await client.getSymbols('umcbl');
|
||||||
|
const bitcoinUSDFuturesRule = symbolRulesResult.data.find(
|
||||||
|
(row) => row.symbol === symbol
|
||||||
|
);
|
||||||
|
console.log('symbol rules: ', bitcoinUSDFuturesRule);
|
||||||
|
if (!bitcoinUSDFuturesRule) {
|
||||||
|
console.error('Failed to get trading rules for ' + symbol);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const order = {
|
||||||
|
marginCoin,
|
||||||
|
orderType: 'market',
|
||||||
|
side: 'open_long',
|
||||||
|
size: bitcoinUSDFuturesRule.minTradeNum,
|
||||||
|
symbol,
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
console.log('placing order: ', order);
|
||||||
|
|
||||||
|
const result = await client.submitOrder(order);
|
||||||
|
|
||||||
|
console.log('order result: ', result);
|
||||||
|
} catch (e) {
|
||||||
|
console.error('request failed: ', e);
|
||||||
|
}
|
||||||
|
})();
|
||||||
104
examples/rest-trade-spot.ts
Normal file
104
examples/rest-trade-spot.ts
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
import { SpotClient, WebsocketClient } from '../src/index';
|
||||||
|
|
||||||
|
// or
|
||||||
|
// import { SpotClient } from 'bitget-api';
|
||||||
|
|
||||||
|
// read from environmental variables
|
||||||
|
const API_KEY = process.env.API_KEY_COM;
|
||||||
|
const API_SECRET = process.env.API_SECRET_COM;
|
||||||
|
const API_PASS = process.env.API_PASS_COM;
|
||||||
|
|
||||||
|
const client = new SpotClient({
|
||||||
|
apiKey: API_KEY,
|
||||||
|
// apiKey: 'apiKeyHere',
|
||||||
|
apiSecret: API_SECRET,
|
||||||
|
// apiSecret: 'apiSecretHere',
|
||||||
|
apiPass: API_PASS,
|
||||||
|
// apiPass: 'apiPassHere',
|
||||||
|
});
|
||||||
|
|
||||||
|
const wsClient = new WebsocketClient({
|
||||||
|
apiKey: API_KEY,
|
||||||
|
apiSecret: API_SECRET,
|
||||||
|
apiPass: API_PASS,
|
||||||
|
});
|
||||||
|
|
||||||
|
function logWSEvent(type, data) {
|
||||||
|
console.log(new Date(), `WS ${type} event: `, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// simple sleep function
|
||||||
|
function promiseSleep(milliseconds) {
|
||||||
|
return new Promise((resolve) => setTimeout(resolve, milliseconds));
|
||||||
|
}
|
||||||
|
|
||||||
|
// WARNING: for sensitive math you should be using a library such as decimal.js!
|
||||||
|
function roundDown(value, decimals) {
|
||||||
|
return Number(
|
||||||
|
Math.floor(parseFloat(value + 'e' + decimals)) + 'e-' + decimals
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** This is a simple script wrapped in a immediately invoked function expression, designed to check for any available BTC balance and immediately sell the full amount for USDT */
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
// Add event listeners to log websocket events on account
|
||||||
|
wsClient.on('update', (data) => logWSEvent('update', data));
|
||||||
|
wsClient.on('open', (data) => logWSEvent('open', data));
|
||||||
|
wsClient.on('response', (data) => logWSEvent('response', data));
|
||||||
|
wsClient.on('reconnect', (data) => logWSEvent('reconnect', data));
|
||||||
|
wsClient.on('reconnected', (data) => logWSEvent('reconnected', data));
|
||||||
|
wsClient.on('authenticated', (data) => logWSEvent('authenticated', data));
|
||||||
|
wsClient.on('exception', (data) => logWSEvent('exception', data));
|
||||||
|
|
||||||
|
// Subscribe to private account topics
|
||||||
|
wsClient.subscribeTopic('SPBL', 'account');
|
||||||
|
wsClient.subscribeTopic('SPBL', 'orders');
|
||||||
|
|
||||||
|
// wait briefly for ws to be ready (could also use the response or authenticated events, to make sure topics are subscribed to before starting)
|
||||||
|
await promiseSleep(2.5 * 1000);
|
||||||
|
|
||||||
|
const balanceResult = await client.getBalance();
|
||||||
|
const allBalances = balanceResult.data;
|
||||||
|
// const balances = allBalances.filter((bal) => Number(bal.available) != 0);
|
||||||
|
const balanceBTC = allBalances.find((bal) => bal.coinName === 'BTC');
|
||||||
|
const btcAmount = balanceBTC ? Number(balanceBTC.available) : 0;
|
||||||
|
// console.log('balance: ', JSON.stringify(balances, null, 2));
|
||||||
|
console.log('BTC balance result: ', balanceBTC);
|
||||||
|
|
||||||
|
if (!btcAmount) {
|
||||||
|
console.error('No BTC to trade');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`BTC available: ${btcAmount}`);
|
||||||
|
const symbol = 'BTCUSDT_SPBL';
|
||||||
|
|
||||||
|
const symbolsResult = await client.getSymbols();
|
||||||
|
const btcRules = symbolsResult.data.find((rule) => rule.symbol === symbol);
|
||||||
|
console.log('btc trading rules: ', btcRules);
|
||||||
|
if (!btcRules) {
|
||||||
|
return console.log('no rules found for trading ' + symbol);
|
||||||
|
}
|
||||||
|
|
||||||
|
const quantityScale = Number(btcRules.quantityScale);
|
||||||
|
// const quantityRoundedDown = btcAmount - btcAmount % 0.01
|
||||||
|
const quantity = roundDown(btcAmount, quantityScale);
|
||||||
|
|
||||||
|
const order = {
|
||||||
|
symbol: symbol,
|
||||||
|
side: 'sell',
|
||||||
|
force: 'normal',
|
||||||
|
orderType: 'market',
|
||||||
|
quantity: String(quantity),
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
console.log('submitting order: ', order);
|
||||||
|
|
||||||
|
const sellResult = await client.submitOrder(order);
|
||||||
|
|
||||||
|
console.log('sell result: ', sellResult);
|
||||||
|
} catch (e) {
|
||||||
|
console.error('request failed: ', e);
|
||||||
|
}
|
||||||
|
})();
|
||||||
@@ -15,6 +15,8 @@ import {
|
|||||||
CancelFuturesPlanTPSL,
|
CancelFuturesPlanTPSL,
|
||||||
HistoricPlanOrderTPSLRequest,
|
HistoricPlanOrderTPSLRequest,
|
||||||
NewFuturesPlanStopOrder,
|
NewFuturesPlanStopOrder,
|
||||||
|
FuturesAccount,
|
||||||
|
FuturesSymbolRule,
|
||||||
} from './types';
|
} from './types';
|
||||||
import { REST_CLIENT_TYPE_ENUM } from './util';
|
import { REST_CLIENT_TYPE_ENUM } from './util';
|
||||||
import BaseRestClient from './util/BaseRestClient';
|
import BaseRestClient from './util/BaseRestClient';
|
||||||
@@ -34,7 +36,9 @@ export class FuturesClient extends BaseRestClient {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/** Get Symbols : Get basic configuration information of all trading pairs (including rules) */
|
/** Get Symbols : Get basic configuration information of all trading pairs (including rules) */
|
||||||
getSymbols(productType: FuturesProductType): Promise<APIResponse<any[]>> {
|
getSymbols(
|
||||||
|
productType: FuturesProductType
|
||||||
|
): Promise<APIResponse<FuturesSymbolRule[]>> {
|
||||||
return this.get('/api/mix/v1/market/contracts', { productType });
|
return this.get('/api/mix/v1/market/contracts', { productType });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -125,7 +129,10 @@ export class FuturesClient extends BaseRestClient {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/** Get Single Account */
|
/** Get Single Account */
|
||||||
getAccount(symbol: string, marginCoin: string): Promise<APIResponse<any>> {
|
getAccount(
|
||||||
|
symbol: string,
|
||||||
|
marginCoin: string
|
||||||
|
): Promise<APIResponse<FuturesAccount>> {
|
||||||
return this.getPrivate('/api/mix/v1/account/account', {
|
return this.getPrivate('/api/mix/v1/account/account', {
|
||||||
symbol,
|
symbol,
|
||||||
marginCoin,
|
marginCoin,
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ import {
|
|||||||
Pagination,
|
Pagination,
|
||||||
APIResponse,
|
APIResponse,
|
||||||
KlineInterval,
|
KlineInterval,
|
||||||
|
CoinBalance,
|
||||||
|
SymbolRules,
|
||||||
} from './types';
|
} from './types';
|
||||||
import { REST_CLIENT_TYPE_ENUM } from './util';
|
import { REST_CLIENT_TYPE_ENUM } from './util';
|
||||||
import BaseRestClient from './util/BaseRestClient';
|
import BaseRestClient from './util/BaseRestClient';
|
||||||
@@ -39,7 +41,7 @@ export class SpotClient extends BaseRestClient {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Get Symbols : Get basic configuration information of all trading pairs (including rules) */
|
/** Get Symbols : Get basic configuration information of all trading pairs (including rules) */
|
||||||
getSymbols(): Promise<APIResponse<any[]>> {
|
getSymbols(): Promise<APIResponse<SymbolRules[]>> {
|
||||||
return this.get('/api/spot/v1/public/products');
|
return this.get('/api/spot/v1/public/products');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -184,7 +186,7 @@ export class SpotClient extends BaseRestClient {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Get Account : get account assets */
|
/** Get Account : get account assets */
|
||||||
getBalance(coin?: string): Promise<APIResponse<any>> {
|
getBalance(coin?: string): Promise<APIResponse<CoinBalance[]>> {
|
||||||
return this.getPrivate('/api/spot/v1/account/assets', { coin });
|
return this.getPrivate('/api/spot/v1/account/assets', { coin });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { OrderTimeInForce } from './shared';
|
||||||
|
|
||||||
export type FuturesProductType =
|
export type FuturesProductType =
|
||||||
| 'umcbl'
|
| 'umcbl'
|
||||||
| 'dmcbl'
|
| 'dmcbl'
|
||||||
@@ -39,7 +41,7 @@ export interface NewFuturesOrder {
|
|||||||
price?: string;
|
price?: string;
|
||||||
side: FuturesOrderSide;
|
side: FuturesOrderSide;
|
||||||
orderType: FuturesOrderType;
|
orderType: FuturesOrderType;
|
||||||
timeInForceValue?: string;
|
timeInForceValue?: OrderTimeInForce;
|
||||||
clientOid?: string;
|
clientOid?: string;
|
||||||
presetTakeProfitPrice?: string;
|
presetTakeProfitPrice?: string;
|
||||||
presetStopLossPrice?: string;
|
presetStopLossPrice?: string;
|
||||||
|
|||||||
@@ -7,3 +7,5 @@ export interface Pagination {
|
|||||||
/** Elements per page */
|
/** Elements per page */
|
||||||
limit?: string;
|
limit?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type OrderTimeInForce = 'normal' | 'post_only' | 'fok' | 'ioc';
|
||||||
|
|||||||
@@ -1,7 +1,4 @@
|
|||||||
import { numberInString, OrderSide } from '../shared';
|
import { OrderTimeInForce } from './shared';
|
||||||
|
|
||||||
export type OrderTypeSpot = 'LIMIT' | 'MARKET' | 'LIMIT_MAKER';
|
|
||||||
export type OrderTimeInForce = 'GTC' | 'FOK' | 'IOC';
|
|
||||||
|
|
||||||
export type WalletType = 'spot' | 'mix_usdt' | 'mix_usd';
|
export type WalletType = 'spot' | 'mix_usdt' | 'mix_usd';
|
||||||
|
|
||||||
@@ -15,9 +12,9 @@ export interface NewWalletTransfer {
|
|||||||
|
|
||||||
export interface NewSpotOrder {
|
export interface NewSpotOrder {
|
||||||
symbol: string;
|
symbol: string;
|
||||||
side: string;
|
side: 'buy' | 'sell';
|
||||||
orderType: string;
|
orderType: 'limit' | 'market';
|
||||||
force: string;
|
force: OrderTimeInForce;
|
||||||
price?: string;
|
price?: string;
|
||||||
quantity: string;
|
quantity: string;
|
||||||
clientOrderId?: string;
|
clientOrderId?: string;
|
||||||
|
|||||||
35
src/types/response/futures.ts
Normal file
35
src/types/response/futures.ts
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
export interface FuturesAccount {
|
||||||
|
marginCoin: string;
|
||||||
|
locked: number;
|
||||||
|
available: number;
|
||||||
|
crossMaxAvailable: number;
|
||||||
|
fixedMaxAvailable: number;
|
||||||
|
maxTransferOut: number;
|
||||||
|
equity: number;
|
||||||
|
usdtEquity: number;
|
||||||
|
btcEquity: number;
|
||||||
|
crossRiskRate: number;
|
||||||
|
crossMarginLeverage: number;
|
||||||
|
fixedLongLeverage: number;
|
||||||
|
fixedShortLeverage: number;
|
||||||
|
marginMode: string;
|
||||||
|
holdMode: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FuturesSymbolRule {
|
||||||
|
baseCoin: string;
|
||||||
|
buyLimitPriceRatio: string;
|
||||||
|
feeRateUpRatio: string;
|
||||||
|
makerFeeRate: string;
|
||||||
|
minTradeNum: string;
|
||||||
|
openCostUpRatio: string;
|
||||||
|
priceEndStep: string;
|
||||||
|
pricePlace: string;
|
||||||
|
quoteCoin: string;
|
||||||
|
sellLimitPriceRatio: string;
|
||||||
|
sizeMultiplier: string;
|
||||||
|
supportMarginCoins: string[];
|
||||||
|
symbol: string;
|
||||||
|
takerFeeRate: string;
|
||||||
|
volumePlace: string;
|
||||||
|
}
|
||||||
@@ -1 +1,3 @@
|
|||||||
export * from './shared';
|
export * from './shared';
|
||||||
|
export * from './spot';
|
||||||
|
export * from './futures';
|
||||||
|
|||||||
22
src/types/response/spot.ts
Normal file
22
src/types/response/spot.ts
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
export interface CoinBalance {
|
||||||
|
coinId: number;
|
||||||
|
coinName: string;
|
||||||
|
available: string;
|
||||||
|
frozen: string;
|
||||||
|
lock: string;
|
||||||
|
uTime: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SymbolRules {
|
||||||
|
symbol: string;
|
||||||
|
symbolName: string;
|
||||||
|
baseCoin: string;
|
||||||
|
quoteCoin: string;
|
||||||
|
minTradeAmount: string;
|
||||||
|
maxTradeAmount: string;
|
||||||
|
takerFeeRate: string;
|
||||||
|
makerFeeRate: string;
|
||||||
|
priceScale: string;
|
||||||
|
quantityScale: string;
|
||||||
|
status: string;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user