diff --git a/README.md b/README.md index 03a63de4..17ce241c 100644 --- a/README.md +++ b/README.md @@ -24,8 +24,6 @@ _Update - Redislabs free tier drops connections very often. Recommend upgrading ## 1-click Installation -_Update - DigitalOcean's app platform is terribly slow. Recommend using render.com for all new installations. All other instructions remain as is._ - [![Deploy to Render](https://render.com/images/deploy-to-render-button.svg)](https://render.com/deploy) or, deploy the application on DigitalOcean's (DO) apps platform. diff --git a/lib/browserUtils.ts b/lib/browserUtils.ts index c76c4642..1e87829a 100644 --- a/lib/browserUtils.ts +++ b/lib/browserUtils.ts @@ -85,7 +85,6 @@ export function commonOnChangeHandler ( const getSchedulingApiProps = ({ isAutoSquareOffEnabled, squareOffTime, - exitStrategy, runAt, runNow, expireIfUnsuccessfulInMins @@ -97,9 +96,7 @@ const getSchedulingApiProps = ({ .format(), autoSquareOffProps: isAutoSquareOffEnabled ? { - time: squareOffTime, - deletePendingOrders: - exitStrategy !== EXIT_STRATEGIES.MULTI_LEG_PREMIUM_THRESHOLD + time: squareOffTime } : undefined, expiresAt: expireIfUnsuccessfulInMins @@ -193,7 +190,6 @@ export const formatFormDataForApi = ({ ...getSchedulingApiProps({ isAutoSquareOffEnabled, squareOffTime, - exitStrategy, expireIfUnsuccessfulInMins, runAt, runNow @@ -237,7 +233,6 @@ export const formatFormDataForApi = ({ ...getSchedulingApiProps({ isAutoSquareOffEnabled, squareOffTime, - exitStrategy, expireIfUnsuccessfulInMins, runAt, runNow diff --git a/lib/constants.ts b/lib/constants.ts index 5ac0d290..767fa474 100644 --- a/lib/constants.ts +++ b/lib/constants.ts @@ -57,7 +57,6 @@ export const INSTRUMENT_DETAILS: Record = { exchange: 'NSE', strikeStepSize: 100, // [27501-40000] - // freezeQty: 100 freezeQty: 1200 }, [INSTRUMENTS.FINNIFTY]: { diff --git a/lib/exit-strategies/autoSquareOff.ts b/lib/exit-strategies/autoSquareOff.ts index 49ac46cd..82b7c853 100644 --- a/lib/exit-strategies/autoSquareOff.ts +++ b/lib/exit-strategies/autoSquareOff.ts @@ -1,4 +1,5 @@ import { KiteOrder } from '../../types/kite' +import { ASO_TYPE } from '../../types/misc' import { ATM_STRADDLE_TRADE, ATM_STRANGLE_TRADE, @@ -7,6 +8,7 @@ import { import { USER_OVERRIDE } from '../constants' import console from '../logging' import { + convertSllToMarketOrder, // logDeep, patchDbTrade, remoteOrderSuccessEnsurer, @@ -122,29 +124,36 @@ export async function doSquareOffPositions ( } async function autoSquareOffStrat ({ + squareOffType, rawKiteOrdersResponse, - deletePendingOrders, initialJobData }: { + squareOffType: ASO_TYPE rawKiteOrdersResponse: KiteOrder[] - deletePendingOrders: boolean initialJobData: SUPPORTED_TRADE_CONFIG }): Promise { const { user } = initialJobData const kite = syncGetKiteInstance(user) - const completedOrders = rawKiteOrdersResponse - - if (deletePendingOrders) { - // console.log('deletePendingOrders init') + if (squareOffType === ASO_TYPE.SLL_TO_MARKET) { + const triggerPendingOrders = rawKiteOrdersResponse + // if completed orders exist, they can be converted to market hours try { - await doDeletePendingOrders(completedOrders, kite) - // console.log('🟢 deletePendingOrders success', res) + await Promise.all( + triggerPendingOrders.map(order => convertSllToMarketOrder(kite, order)) + ) } catch (e) { console.log('🔴 deletePendingOrders failed') console.error(e) } + } else { + const completedOrders = rawKiteOrdersResponse + try { + await doSquareOffPositions(completedOrders, kite, initialJobData) + } catch (e) { + console.log('🔴 doSquareOffPositions failed') + console.error(e) + } } - return doSquareOffPositions(completedOrders, kite, initialJobData) } export default autoSquareOffStrat diff --git a/lib/exit-strategies/individualLegExitOrders.ts b/lib/exit-strategies/individualLegExitOrders.ts index 1206ed02..0dd2adfb 100644 --- a/lib/exit-strategies/individualLegExitOrders.ts +++ b/lib/exit-strategies/individualLegExitOrders.ts @@ -1,12 +1,17 @@ import { KiteOrder } from '../../types/kite' +import { ASO_TYPE } from '../../types/misc' import { SL_ORDER_TYPE } from '../../types/plans' import { SUPPORTED_TRADE_CONFIG } from '../../types/trade' import console from '../logging' -import { addToNextQueue, WATCHER_Q_NAME } from '../queue' -import orderResponse from '../strategies/mockData/orderResponse' +import { + addToAutoSquareOffQueue, + addToNextQueue, + WATCHER_Q_NAME +} from '../queue' +// import orderResponse from '../strategies/mockData/orderResponse' import { attemptBrokerOrders, - isUntestedFeaturesEnabled, + // isUntestedFeaturesEnabled, remoteOrderSuccessEnsurer, round, syncGetKiteInstance @@ -62,10 +67,10 @@ async function individualLegExitOrders ({ orderTag, rollback, slLimitPricePercent = 1, - instrument + instrument, + isAutoSquareOffEnabled } = initialJobData - const slOrderType = SL_ORDER_TYPE.SLL const kite = _kite || syncGetKiteInstance(user) const exitOrders = completedOrders.map(order => { @@ -103,10 +108,7 @@ async function individualLegExitOrders ({ exchange } - if (slOrderType === SL_ORDER_TYPE.SLL) { - exitOrder = convertSlmToSll(exitOrder, slLimitPricePercent!, kite) - } - + exitOrder = convertSlmToSll(exitOrder, slLimitPricePercent!, kite) exitOrder.trigger_price = round(exitOrder.trigger_price!) console.log('placing exit orders...', exitOrder) return exitOrder @@ -132,20 +134,26 @@ async function individualLegExitOrders ({ throw Error('rolled back onBrokenExitOrders') } - if (slOrderType === SL_ORDER_TYPE.SLL) { - const watcherQueueJobs = statefulOrders.map(async exitOrder => { - return addToNextQueue(initialJobData, { - _nextTradingQueue: WATCHER_Q_NAME, - rawKiteOrderResponse: exitOrder - }) + if (isAutoSquareOffEnabled) { + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.SLL_TO_MARKET, + jobResponse: { rawKiteOrdersResponse: statefulOrders }, + initialJobData + }) + } + + const watcherQueueJobs = statefulOrders.map(async exitOrder => { + return addToNextQueue(initialJobData, { + _nextTradingQueue: WATCHER_Q_NAME, + rawKiteOrderResponse: exitOrder }) + }) - try { - await Promise.all(watcherQueueJobs) - } catch (e) { - console.log('error adding to `watcherQueueJobs`') - console.log(e.message ? e.message : e) - } + try { + await Promise.all(watcherQueueJobs) + } catch (e) { + console.log('error adding to `watcherQueueJobs`') + console.log(e.message ? e.message : e) } return statefulOrders diff --git a/lib/exit-strategies/minXPercentOrSupertrend.ts b/lib/exit-strategies/minXPercentOrSupertrend.ts index 0d37ab3f..2919e38c 100644 --- a/lib/exit-strategies/minXPercentOrSupertrend.ts +++ b/lib/exit-strategies/minXPercentOrSupertrend.ts @@ -1,7 +1,6 @@ import axios from 'axios' import dayjs from 'dayjs' import { KiteOrder } from '../../types/kite' -import { SL_ORDER_TYPE } from '../../types/plans' import { DIRECTIONAL_OPTION_SELLING_TRADE } from '../../types/trade' import console from '../logging' @@ -10,6 +9,7 @@ import { attemptBrokerOrders, getLastOpenDateSince, getNearestCandleTime, + getOrderHistory, getPercentageChange, logDeep, remoteOrderSuccessEnsurer, @@ -35,15 +35,14 @@ async function minXPercentOrSupertrend ({ hedgeOrderResponse }: DOS_TRAILING_INTERFACE): Promise { const { user, orderTag, slLimitPricePercent = 1, instrument } = initialJobData - const slOrderType = SL_ORDER_TYPE.SLL try { const kite = syncGetKiteInstance(user) const [rawKiteOrderResponse] = rawKiteOrdersResponse // NB: rawKiteOrderResponse here is of pending SLM Order - const orderHistory: KiteOrder[] = await withRemoteRetry(() => - kite.getOrderHistory(rawKiteOrderResponse.order_id) + const byRecencyOrderHistory: KiteOrder[] = await getOrderHistory( + kite, + rawKiteOrderResponse.order_id! ) - const byRecencyOrderHistory = orderHistory.reverse() const isSlOrderCancelled = byRecencyOrderHistory.find( odr => odr.status === 'CANCELLED' @@ -126,28 +125,21 @@ async function minXPercentOrSupertrend ({ // console.log('should trail SL!') try { - const sllOrderProps = - slOrderType === SL_ORDER_TYPE.SLL - ? convertSlmToSll( - { - transaction_type: kite.TRANSACTION_TYPE_BUY, - trigger_price: newSL - } as KiteOrder, - slLimitPricePercent!, - kite - ) - : null + const sllOrderProps = convertSlmToSll( + { + transaction_type: kite.TRANSACTION_TYPE_BUY, + trigger_price: newSL + } as KiteOrder, + slLimitPricePercent!, + kite + ) const res = await kite.modifyOrder( triggerPendingOrder!.variety, triggerPendingOrder!.order_id, - slOrderType === SL_ORDER_TYPE.SLL - ? { - trigger_price: sllOrderProps!.trigger_price, - price: sllOrderProps!.price - } - : { - trigger_price: newSL - } + { + trigger_price: sllOrderProps!.trigger_price, + price: sllOrderProps!.price + } ) console.log( `🟢 [minXPercentOrSupertrend] SL modified from ${String( @@ -175,10 +167,7 @@ async function minXPercentOrSupertrend ({ tag: orderTag! } - if (slOrderType === SL_ORDER_TYPE.SLL) { - exitOrder = convertSlmToSll(exitOrder, slLimitPricePercent!, kite) - } - + exitOrder = convertSlmToSll(exitOrder, slLimitPricePercent!, kite) const remoteOrder = remoteOrderSuccessEnsurer({ ensureOrderState: 'TRIGGER PENDING', instrument, diff --git a/lib/exit-strategies/multiLegPremiumThreshold.ts b/lib/exit-strategies/multiLegPremiumThreshold.ts index 99ab510a..f9f2f69f 100644 --- a/lib/exit-strategies/multiLegPremiumThreshold.ts +++ b/lib/exit-strategies/multiLegPremiumThreshold.ts @@ -39,24 +39,20 @@ import { syncGetKiteInstance, withRemoteRetry, patchDbTrade, - getMultipleInstrumentPrices + getMultipleInstrumentPrices, + logDeep } from '../utils' import { doSquareOffPositions } from './autoSquareOff' -const patchTradeWithTrailingSL = async ({ dbId, trailingSl }) => { - try { - await patchDbTrade({ - _id: dbId, - patchProps: { - liveTrailingSl: trailingSl, - lastTrailingSlSetAt: dayjs().format() - } - }) - } catch (e) { - console.log('🔴 [patchTradeWithTrailingSL] error', e) - } -} +const patchTradeWithTrailingSL = async ({ dbId, trailingSl }) => + await patchDbTrade({ + _id: dbId, + patchProps: { + liveTrailingSl: trailingSl, + lastTrailingSlSetAt: dayjs().format() + } + }) const tradeHeartbeat = async dbId => { const data = await patchDbTrade({ @@ -144,7 +140,14 @@ async function multiLegPremiumThreshold ({ '🔴 [multiLegPremiumThreshold] getInstrumentPrice error', error ) - return Promise.reject(new Error('Kite APIs acting up')) + // [TODO] see if we can resolve this and add back to the queue to prevent memory leak issues + await addToNextQueue(initialJobData, { + _nextTradingQueue: EXIT_TRADING_Q_NAME, + rawKiteOrdersResponse, + squareOffOrders + }) + + return Promise.resolve('Kite APIs acting up!') } const liveTotalPremium = tradingSymbols.reduce((sum, tradingSymbol) => { @@ -226,7 +229,12 @@ async function multiLegPremiumThreshold ({ if (liveTotalPremium < checkAgainstSl) { const rejectMsg = `🟢 [multiLegPremiumThreshold] liveTotalPremium (${liveTotalPremium}) < threshold (${checkAgainstSl})` - return Promise.reject(new Error(rejectMsg)) + await addToNextQueue(initialJobData, { + _nextTradingQueue: EXIT_TRADING_Q_NAME, + rawKiteOrdersResponse, + squareOffOrders + }) + return Promise.resolve(rejectMsg) } // terminate the checker @@ -280,13 +288,13 @@ async function multiLegPremiumThreshold ({ legOrder => legOrder.tradingsymbol === losingLeg.tradingSymbol ) ) - // console.log('squareOffLosingLegs', logDeep(squareOffLosingLegs)) + console.log('squareOffLosingLegs', logDeep(squareOffLosingLegs)) const bringToCostOrders = winningLegs.map(winningLeg => legsOrders.find( legOrder => legOrder.tradingsymbol === winningLeg.tradingSymbol ) ) - // console.log('bringToCostOrders', logDeep(bringToCostOrders)) + console.log('bringToCostOrders', logDeep(bringToCostOrders)) // 1. square off losing legs await doSquareOffPositions( squareOffLosingLegs as KiteOrder[], diff --git a/lib/queue-processor/exitTradingQueue.ts b/lib/queue-processor/exitTradingQueue.ts index 12aa51a5..f01f300f 100644 --- a/lib/queue-processor/exitTradingQueue.ts +++ b/lib/queue-processor/exitTradingQueue.ts @@ -66,7 +66,7 @@ const worker = new Worker( const exitOrders = await processJob(job.data) return exitOrders } catch (e) { - console.log(e.message ? e.message : e) + // console.log(e.message ? e.message : e) throw new Error(e) } }, @@ -85,10 +85,11 @@ worker.on('error', err => { console.log('🔴 [exitTradingQueue] worker error', err) }) -// worker.on('completed', (job) => { -// // const { id, name } = job -// // console.log('// job has completed', { id, name }) -// }) +worker.on('completed', async job => { + await job.remove() + // const { id, name } = job + // console.log('// job has completed', { id, name }) +}) // worker.on('failed', (job) => { // try { diff --git a/lib/queue-processor/optionSellerStrategy/optionEntryWatcher.js b/lib/queue-processor/optionSellerStrategy/optionEntryWatcher.js index 6051c2c9..848f9975 100644 --- a/lib/queue-processor/optionSellerStrategy/optionEntryWatcher.js +++ b/lib/queue-processor/optionSellerStrategy/optionEntryWatcher.js @@ -1,5 +1,6 @@ import { INSTRUMENT_DETAILS } from '../../constants' import { + getCompletedOrderFromOrderHistoryById, getIndexInstruments, getInstrumentPrice, getTradingSymbolsByOptionPrice, @@ -20,10 +21,9 @@ const optionSellerEntryWatcher = async ({ try { const { user, orderTag, instrument, expiryType } = initialJobData const kite = syncGetKiteInstance(user) - const orderHistory = await kite.getOrderHistory(limitOrderAckId) - const revOrderHistory = orderHistory.reverse() - const completedOrder = revOrderHistory.find( - order => order.status === kite.STATUS_COMPLETE + const completedOrder = await getCompletedOrderFromOrderHistoryById( + kite, + limitOrderAckId ) if (!completedOrder) { return Promise.reject( diff --git a/lib/queue-processor/optionSellerStrategy/optionSLWatcher.js b/lib/queue-processor/optionSellerStrategy/optionSLWatcher.js index 3ad7806d..dc30f190 100644 --- a/lib/queue-processor/optionSellerStrategy/optionSLWatcher.js +++ b/lib/queue-processor/optionSellerStrategy/optionSLWatcher.js @@ -1,4 +1,8 @@ -import { syncGetKiteInstance } from '../../utils' +import { + getCompletedOrderFromOrderHistoryById, + getOrderHistory, + syncGetKiteInstance +} from '../../utils' import console from '../../logging' @@ -14,10 +18,9 @@ const optionSellerSLWatcher = async ({ try { const { user, orderTag } = initialJobData const kite = syncGetKiteInstance(user) - const orderHistory = await kite.getOrderHistory(slOrderAckId) - const revOrderHistory = orderHistory.reverse() - const completedOrder = revOrderHistory.find( - order => order.status === kite.STATUS_COMPLETE + const completedOrder = await getCompletedOrderFromOrderHistoryById( + kite, + slOrderAckId ) if (!completedOrder) { return Promise.reject(new Error('[optionSellerSLWatcher] still pending!')) diff --git a/lib/queue-processor/tradingQueue.ts b/lib/queue-processor/tradingQueue.ts index 381e6539..cefe1015 100644 --- a/lib/queue-processor/tradingQueue.ts +++ b/lib/queue-processor/tradingQueue.ts @@ -4,7 +4,6 @@ import { Job, Worker } from 'bullmq' import { ANCILLARY_TASKS, STRATEGIES } from '../constants' import console from '../logging' import { - addToAutoSquareOffQueue, addToNextQueue, ANCILLARY_Q_NAME, redisConnection, @@ -65,27 +64,6 @@ const worker = new Worker( } catch (e) { console.log('[error] enabling orderbook sync by tag...', e) } - - const { isAutoSquareOffEnabled, strategy } = job.data - // can't enable auto square off for DOS - // because we don't know upfront how many orders would get punched - if ( - strategy !== STRATEGIES.DIRECTIONAL_OPTION_SELLING && - isAutoSquareOffEnabled - ) { - try { - // console.log('enabling auto square off...') - const asoResponse = await addToAutoSquareOffQueue({ - //eslint-disable-line - initialJobData: job.data, - jobResponse: result - }) - // const { data, name } = asoResponse - // console.log('🟢 success enable auto square off', { data, name }) - } catch (e) { - console.log('🔴 failed to enable auto square off', e) - } - } return result }, { diff --git a/lib/queue.ts b/lib/queue.ts index c65ad9fa..1b02de3e 100644 --- a/lib/queue.ts +++ b/lib/queue.ts @@ -2,6 +2,9 @@ import { Queue, QueueScheduler, JobsOptions, Job } from 'bullmq' import dayjs from 'dayjs' import IORedis from 'ioredis' import { v4 as uuidv4 } from 'uuid' +import { KiteOrder } from '../types/kite' +import { ASO_TYPE } from '../types/misc' +import { SUPPORTED_TRADE_CONFIG } from '../types/trade' import console from './logging' import { @@ -11,6 +14,7 @@ import { getQueueOptionsForExitStrategy, getTimeLeftInMarketClosingMs, isMockOrder, + logDeep, ms } from './utils' @@ -165,12 +169,19 @@ export async function addToNextQueue ( } export async function addToAutoSquareOffQueue ({ + squareOffType, initialJobData, jobResponse -}) { - const { - autoSquareOffProps: { time, deletePendingOrders } - } = initialJobData +}: { + squareOffType: ASO_TYPE + jobResponse: { + rawKiteOrdersResponse: KiteOrder[] + squareOffOrders?: KiteOrder[] + } + initialJobData: SUPPORTED_TRADE_CONFIG +}): Promise { + const { autoSquareOffProps } = initialJobData + const { time } = autoSquareOffProps as { time: string } const { rawKiteOrdersResponse, squareOffOrders } = jobResponse const finalOrderTime = getMisOrderLastSquareOffTime() const runAtTime = isMockOrder() @@ -180,19 +191,25 @@ export async function addToAutoSquareOffQueue ({ : time const delay = dayjs(runAtTime).diff(dayjs()) - // console.log(`>>> auto square off scheduled for ${Math.ceil(delay / 60000)} minutes from now`) + console.log( + `>>> auto square off scheduled for ${Math.ceil( + delay / 60000 + )} minutes from now` + ) + const queueProps = { + squareOffType, + rawKiteOrdersResponse: squareOffOrders || rawKiteOrdersResponse, + initialJobData + } + logDeep(queueProps) return autoSquareOffQueue.add( `${AUTO_SQUARE_OFF_Q_NAME}_${uuidv4() as string}`, - { - rawKiteOrdersResponse: squareOffOrders || rawKiteOrdersResponse, - deletePendingOrders, - initialJobData - }, + queueProps, { delay } ) } -export const cleanupQueues = async () => +export const cleanupQueues = async (): Promise => await Promise.all(allQueues.map(async queue => await queue.obliterate())) diff --git a/lib/strategies/atmStraddle.ts b/lib/strategies/atmStraddle.ts index 1fb39ebe..c24bcf84 100644 --- a/lib/strategies/atmStraddle.ts +++ b/lib/strategies/atmStraddle.ts @@ -1,10 +1,11 @@ import dayjs, { ConfigType } from 'dayjs' import { KiteOrder } from '../../types/kite' -import { SignalXUser } from '../../types/misc' +import { SignalXUser, ASO_TYPE } from '../../types/misc' import { ATM_STRADDLE_TRADE } from '../../types/trade' import { EXPIRY_TYPE, + EXIT_STRATEGIES, INSTRUMENT_DETAILS, INSTRUMENT_PROPERTIES, PRODUCT_TYPE, @@ -12,7 +13,7 @@ import { } from '../constants' import { doSquareOffPositions } from '../exit-strategies/autoSquareOff' import console from '../logging' -import { EXIT_TRADING_Q_NAME } from '../queue' +import { EXIT_TRADING_Q_NAME, addToAutoSquareOffQueue } from '../queue' import { attemptBrokerOrders, delay, @@ -193,24 +194,9 @@ export const createOrder = ({ } } -async function atmStraddle ({ - _kite, - instrument, - lots, - user, - expiresAt, - orderTag, - rollback, - maxSkewPercent, - thresholdSkewPercent, - takeTradeIrrespectiveSkew, - isHedgeEnabled, - hedgeDistance, - productType = PRODUCT_TYPE.MIS, - volatilityType = VOLATILITY_TYPE.SHORT, - expiryType = EXPIRY_TYPE.CURRENT, - _nextTradingQueue = EXIT_TRADING_Q_NAME -}: ATM_STRADDLE_TRADE): Promise< +async function atmStraddle ( + jobData: ATM_STRADDLE_TRADE +): Promise< | { _nextTradingQueue: string straddle: Record @@ -219,6 +205,26 @@ async function atmStraddle ({ } | undefined > { + const { + _kite, + instrument, + lots, + user, + expiresAt, + orderTag, + rollback, + maxSkewPercent, + thresholdSkewPercent, + takeTradeIrrespectiveSkew, + isAutoSquareOffEnabled, + isHedgeEnabled, + hedgeDistance, + exitStrategy, + productType = PRODUCT_TYPE.MIS, + volatilityType = VOLATILITY_TYPE.SHORT, + expiryType = EXPIRY_TYPE.CURRENT, + _nextTradingQueue = EXIT_TRADING_Q_NAME + } = jobData const kite = _kite || syncGetKiteInstance(user) const { @@ -252,6 +258,7 @@ async function atmStraddle ({ let allOrdersLocal: KiteOrder[] = [] let hedgeOrdersLocal: KiteOrder[] = [] + let hedgeOrders: KiteOrder[] = [] let allOrders: KiteOrder[] = [] if (volatilityType === VOLATILITY_TYPE.SHORT && isHedgeEnabled) { @@ -324,6 +331,7 @@ async function atmStraddle ({ throw Error('rolled back onBrokenHedgeOrders') } + hedgeOrders = [...statefulOrders] allOrders = [...statefulOrders] } @@ -347,6 +355,27 @@ async function atmStraddle ({ throw Error('rolled back on onBrokenPrimaryOrders') } + if (isAutoSquareOffEnabled) { + if (exitStrategy === EXIT_STRATEGIES.MULTI_LEG_PREMIUM_THRESHOLD) { + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.OPEN_POSITION_SQUARE_OFF, + jobResponse: { rawKiteOrdersResponse: allOrders }, + initialJobData: jobData + }) + } else { + // if hedges are enabled, + // turn on auto square off on those positions + // as they won't have SL orders + if (hedgeOrders.length) { + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.OPEN_POSITION_SQUARE_OFF, + jobResponse: { rawKiteOrdersResponse: hedgeOrders }, + initialJobData: jobData + }) + } + } + } + return { _nextTradingQueue, straddle, diff --git a/lib/strategies/directionalOptionSelling.ts b/lib/strategies/directionalOptionSelling.ts index d0089e61..45ec7ead 100644 --- a/lib/strategies/directionalOptionSelling.ts +++ b/lib/strategies/directionalOptionSelling.ts @@ -36,6 +36,7 @@ import { withRemoteRetry, logDeep } from '../utils' +import { ASO_TYPE } from '../../types/misc' const SIGNALX_URL = process.env.SIGNALX_URL ?? 'https://indicator.signalx.trade' @@ -436,16 +437,25 @@ async function punchOrders ( ].filter(o => o) if (isAutoSquareOffEnabled) { try { - const asoResponse = await addToAutoSquareOffQueue({ + if (hedgeOrdersResponse.length) { + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.OPEN_POSITION_SQUARE_OFF, + initialJobData: nextQueueData, + jobResponse: { + rawKiteOrdersResponse: hedgeOrdersResponse + } + }) + } + + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.SLL_TO_MARKET, initialJobData: nextQueueData, jobResponse: { - rawKiteOrdersResponse: allPunchedOrders + rawKiteOrdersResponse: exitOrders } }) - const { data, name } = asoResponse console.log( - '🟢 [directionalOptionSelling] success enable auto square off', - { data, name } + '🟢 [directionalOptionSelling] success enable auto square off' ) } catch (e) { console.log( diff --git a/lib/strategies/strangle.ts b/lib/strategies/strangle.ts index dbdcf680..8bf68cee 100644 --- a/lib/strategies/strangle.ts +++ b/lib/strategies/strangle.ts @@ -6,10 +6,11 @@ import { INSTRUMENT_DETAILS, PRODUCT_TYPE, STRANGLE_ENTRY_STRATEGIES, - VOLATILITY_TYPE + VOLATILITY_TYPE, + EXIT_STRATEGIES } from '../constants' import console from '../logging' -import { EXIT_TRADING_Q_NAME } from '../queue' +import { addToAutoSquareOffQueue, EXIT_TRADING_Q_NAME } from '../queue' import { apiResponseObject, attemptBrokerOrders, @@ -28,6 +29,7 @@ import { doSquareOffPositions } from '../exit-strategies/autoSquareOff' import dayjs, { Dayjs } from 'dayjs' import { KiteOrder } from '../../types/kite' import axios from 'axios' +import { ASO_TYPE } from '../../types/misc' export const getNearestContractDate = async ( atmStrike: number, @@ -140,7 +142,7 @@ const getStrangleStrikes = async ({ } } -async function atmStrangle (args: ATM_STRANGLE_TRADE) { +async function atmStrangle (jobData: ATM_STRANGLE_TRADE) { try { const { instrument, @@ -157,8 +159,10 @@ async function atmStrangle (args: ATM_STRANGLE_TRADE) { productType = PRODUCT_TYPE.MIS, volatilityType = VOLATILITY_TYPE.SHORT, expiryType, + isAutoSquareOffEnabled, + exitStrategy, _nextTradingQueue = EXIT_TRADING_Q_NAME - } = args + } = jobData const { lotSize, nfoSymbol, @@ -170,7 +174,7 @@ async function atmStrangle (args: ATM_STRANGLE_TRADE) { const sourceData = await getIndexInstruments() const { atmStrike } = await getATMStrikes({ - ...args, + ...jobData, takeTradeIrrespectiveSkew: true, instrumentsData: sourceData, startTime: dayjs(), @@ -203,6 +207,7 @@ async function atmStrangle (args: ATM_STRANGLE_TRADE) { let allOrdersLocal: KiteOrder[] = [] let hedgeOrdersLocal: KiteOrder[] = [] + let hedgeOrders: KiteOrder[] = [] let allOrders: KiteOrder[] = [] if (volatilityType === VOLATILITY_TYPE.SHORT && isHedgeEnabled) { @@ -278,6 +283,7 @@ async function atmStrangle (args: ATM_STRANGLE_TRADE) { throw Error('rolled back onBrokenHedgeOrders') } + hedgeOrders = [...statefulOrders] allOrders = [...statefulOrders] } @@ -301,6 +307,27 @@ async function atmStrangle (args: ATM_STRANGLE_TRADE) { throw Error('rolled back on onBrokenPrimaryOrders') } + if (isAutoSquareOffEnabled) { + if (exitStrategy === EXIT_STRATEGIES.MULTI_LEG_PREMIUM_THRESHOLD) { + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.OPEN_POSITION_SQUARE_OFF, + jobResponse: { rawKiteOrdersResponse: allOrders }, + initialJobData: jobData + }) + } else { + // all hedges are enabled, + // turn on auto square off on those positions + // as they won't have SL orders + if (hedgeOrders.length) { + await addToAutoSquareOffQueue({ + squareOffType: ASO_TYPE.OPEN_POSITION_SQUARE_OFF, + jobResponse: { rawKiteOrdersResponse: hedgeOrders }, + initialJobData: jobData + }) + } + } + } + return { _nextTradingQueue, rawKiteOrdersResponse: statefulOrders, diff --git a/lib/utils.ts b/lib/utils.ts index f8528946..f8a6ef1c 100644 --- a/lib/utils.ts +++ b/lib/utils.ts @@ -357,8 +357,19 @@ export function syncGetKiteInstance (user) { }) } -export async function getCompletedOrderFromOrderHistoryById (kite, orderId) { - const orders = await kite.getOrderHistory(orderId) +export async function getOrderHistory ( + kite: any, + orderId: string +): Promise { + const history = await withRemoteRetry(() => kite.getOrderHistory(orderId)) + return [...history].reverse() +} + +export async function getCompletedOrderFromOrderHistoryById ( + kite: any, + orderId: string +): Promise { + const orders = await getOrderHistory(kite, orderId) return orders.find(odr => odr.status === 'COMPLETE') } @@ -544,7 +555,20 @@ const marketHolidays = [ ['October 15,2021', 'Friday'], ['November 04,2021', 'Thursday'], ['November 05,2021', 'Friday'], - ['November 19,2021', 'Friday'] + ['November 19,2021', 'Friday'], + ['January 26,2022', 'Wednesday'], + ['March 01,2022', 'Tuesday'], + ['March 18,2022', 'Friday'], + ['April 14,2022', 'Thursday'], + ['April 15,2022', 'Friday'], + ['May 03,2022', 'Tuesday'], + ['August 09,2022', 'Tuesday'], + ['August 15,2022', 'Monday'], + ['August 31,2022', 'Wednesday'], + ['October 05,2022', 'Wednesday'], + ['October 24,2022', 'Monday'], + ['October 26,2022', 'Wednesday'], + ['November 08,2022', 'Tuesday'] ] export const isDateHoliday = (date: Dayjs) => { @@ -945,10 +969,7 @@ export const orderStateChecker = (kite, orderId, ensureOrderState) => { return false } try { - const orderHistory = await withRemoteRetry(() => - kite.getOrderHistory(orderId) - ) - const byRecencyOrderHistory = orderHistory.reverse() + const byRecencyOrderHistory = await getOrderHistory(kite, orderId) // if it reaches here, then order exists in broker system const expectedStateOrder = byRecencyOrderHistory.find( @@ -963,7 +984,6 @@ export const orderStateChecker = (kite, orderId, ensureOrderState) => { orderId, ensureOrderState }) - logDeep(orderHistory) const wasOrderRejectedOrCancelled = byRecencyOrderHistory.find( odr => @@ -1385,3 +1405,20 @@ export function round (value: number, step = 0.5): number { const inv = 1.0 / step return Math.round(value * inv) / inv } + +export const convertSllToMarketOrder = async ( + kite: any, + order: KiteOrder +): Promise => { + // ensure order is not in complete state + const completedOrder = await getCompletedOrderFromOrderHistoryById(kite, order.order_id!) + if (completedOrder) { + console.log(`order #${order.order_id} convertSllToMarketOrder already completed`) + return + } + return withRemoteRetry(() => + kite.modifyOrder(order.variety, order.order_id, { + order_type: kite.ORDER_TYPE_MARKET + }) + ) +} diff --git a/lib/watchers/sllWatcher.ts b/lib/watchers/sllWatcher.ts index 5962ce0c..0249c48b 100644 --- a/lib/watchers/sllWatcher.ts +++ b/lib/watchers/sllWatcher.ts @@ -12,11 +12,12 @@ import { Promise } from 'bluebird' import { SignalXUser } from '../../types/misc' import console from '../logging' import { + convertSllToMarketOrder, finiteStateChecker, + getOrderHistory, ms, orderStateChecker, - syncGetKiteInstance, - withRemoteRetry + syncGetKiteInstance } from '../utils' const sllWatcher = async ({ @@ -28,9 +29,7 @@ const sllWatcher = async ({ }) => { try { const kite = syncGetKiteInstance(user) - const orderHistory = ( - await withRemoteRetry(() => kite.getOrderHistory(sllOrderId)) - ).reverse() + const orderHistory = await getOrderHistory(kite, sllOrderId) const isOrderCompleted = orderHistory.find( order => order.status === kite.STATUS_COMPLETE ) @@ -39,7 +38,7 @@ const sllWatcher = async ({ } const cancelledOrder = orderHistory.find(order => - order.status.includes(kite.STATUS_CANCELLED) + order.status!.includes(kite.STATUS_CANCELLED) ) if (cancelledOrder) { @@ -71,11 +70,7 @@ const sllWatcher = async ({ sllOrderId ) try { - await withRemoteRetry(() => - kite.modifyOrder(openOrder.variety, sllOrderId, { - order_type: kite.ORDER_TYPE_MARKET - }) - ) + await convertSllToMarketOrder(kite, openOrder) return Promise.resolve( `🟢 [sllWatcher] squared off open SLL order id ${sllOrderId}` ) diff --git a/lib/watchers/slmWatcher.ts b/lib/watchers/slmWatcher.ts deleted file mode 100644 index 5c61981b..00000000 --- a/lib/watchers/slmWatcher.ts +++ /dev/null @@ -1,211 +0,0 @@ -/** - * what happens - Exchange cancels orders that lie outside execution range - * - * 1. SLM order can be partially filled before it gets cancelled - * 2. Entire order can be cancelled - * - * Action plan: - * - * 1. Have a reference to the order id created by zerodha - * 2. Every 5 seconds - * 2. SLM order is in state `CANCELLED` or `COMPLETED` - * 3. Cancel checker if `COMPLETED` - * 4. Square off open position qty that was managed by this order_id - * 5. If `Cancelled`, get the order history of this order_id, - * 1. Get the item with status Cancelled. - * 2. Fetch its cancelled_quantity - * 3. Place a market exit order for cancelled_quantity for the tradingsymbol - * 4. Add this new order back to this queue for monitoring - * - */ - -import { KiteOrder } from '../../types/kite' -import { SignalXUser } from '../../types/misc' -import { SUPPORTED_TRADE_CONFIG } from '../../types/trade' -import console from '../logging' -import { addToNextQueue, WATCHER_Q_NAME } from '../queue' -import { - getInstrumentPrice, - remoteOrderSuccessEnsurer, - syncGetKiteInstance, - withRemoteRetry -} from '../utils' - -/** - * [NB] IMPORTANT! - * WATCH_MANUAL_CANCELLED_ORDERS is for testing this only! - * DO NOT enable this env variable on your account! - * e.g. in DOS, Khaching can itself cancel a pending order and create a new SLM order - * if you were to enable this, - * the position will get auto squared off as soon as Khaching cancels that pending order - */ -const WATCH_MANUAL_CANCELLED_ORDERS = process.env.WATCH_MANUAL_CANCELLED_ORDERS - ? JSON.parse(process.env.WATCH_MANUAL_CANCELLED_ORDERS) - : false - -const slmWatcher = async ({ - slmOrderId, - user, - originalTriggerPrice, - _queueJobData -}: { - slmOrderId: string - user: SignalXUser - originalTriggerPrice: number - _queueJobData: { initialJobData: SUPPORTED_TRADE_CONFIG } -}) => { - /** - * Scenario for the need of `originalTriggerPrice` - * - consider the initial SLM order gets cancelled at 50 - * - watcher sees new LTP to be 51 - * - watcher places `market` exit order, but order gets rejected - * - now the ref to slmOrderId will contain no triggerPrice as it was a market order - * - * - * hence - - * - we'll need to pass through the original trigger price down the rabbit hole - */ - try { - const kite = syncGetKiteInstance(user) - const orderHistory = ( - await withRemoteRetry(() => kite.getOrderHistory(slmOrderId)) - ).reverse() - const isOrderCompleted = orderHistory.find( - order => order.status === kite.STATUS_COMPLETE - ) - if (isOrderCompleted) { - return Promise.resolve('[slmWatcher] order COMPLETED!') - } - - const cancelledOrder = orderHistory.find(order => - order.status.includes(kite.STATUS_CANCELLED) - ) - - if (!cancelledOrder) { - return Promise.reject( - new Error('[slmWatcher] neither COMPLETED nor CANCELLED. Watching!') - ) - } - - const { - cancelled_quantity: cancelledQty, - status_message_raw: statusMessageRaw, - transaction_type: transactionType, - trigger_price: cancelledTriggerPrice, - tradingsymbol, - exchange, - product - } = cancelledOrder - - const triggerPrice = originalTriggerPrice || cancelledTriggerPrice - - /** - * Conditions: - * 1. WATCH_MANUAL_CANCELLED_ORDERS = false && statusMessageRaw = null - * true && true - returned - * - * 2. WATCH_MANUAL_CANCELLED_ORDERS = false && statusMessageRaw = '17070' - * true && false - continue - * - * 3. WATCH_MANUAL_CANCELLED_ORDERS = true && statusMessageRaw = null - * false && true - continue - * - * 4. WATCH_MANUAL_CANCELLED_ORDERS = true && statusMessageRaw = '17070' - * false && false - continue - */ - - if ( - !WATCH_MANUAL_CANCELLED_ORDERS && - statusMessageRaw !== - '17070 : The Price is out of the current execution range' - ) { - return Promise.resolve('[slmWatcher] order cancelled by user!') - } - - console.log('🟢 [slmWatcher] found cancelled SLM order!', { - slmOrderId, - cancelledQty, - statusMessageRaw - }) - - if (!cancelledQty) { - return Promise.resolve('[slmWatcher] no cancelled qty!') - } - - const positions = await withRemoteRetry(() => kite.getPositions()) - - const { net } = positions - const openPositionThatMustBeSquaredOff = net.find( - position => - position.tradingsymbol === tradingsymbol && - position.product === product && - position.exchange === exchange && - Math.abs(position.quantity) >= cancelledQty - ) - - if (!openPositionThatMustBeSquaredOff) { - return Promise.resolve('[slmWatcher] no open position to be squared off!') - } - - console.log( - '[slmWatcher] openPositionThatMustBeSquaredOff', - openPositionThatMustBeSquaredOff - ) - - const ltp = await withRemoteRetry(async () => - getInstrumentPrice(kite, tradingsymbol, exchange) - ) - - const newOrderType = - (transactionType === kite.TRANSACTION_TYPE_BUY && ltp < triggerPrice) || - (transactionType === kite.TRANSACTION_TYPE_SELL && ltp > triggerPrice) - ? kite.ORDER_TYPE_SLM - : kite.ORDER_TYPE_MARKET - - const exitOrder: KiteOrder = { - tradingsymbol, - exchange, - product, - quantity: cancelledQty, - transaction_type: transactionType, - order_type: newOrderType, - tag: _queueJobData.initialJobData.orderTag! - } - - if (newOrderType === kite.ORDER_TYPE_SLM) { - exitOrder.trigger_price = triggerPrice - } - - console.log('[slmWatcher] placing exit order', exitOrder) - try { - const { response } = await remoteOrderSuccessEnsurer({ - ensureOrderState: exitOrder.trigger_price - ? 'TRIGGER PENDING' - : kite.STATUS_COMPLETE, - orderProps: exitOrder, - instrument: _queueJobData.initialJobData.instrument, - user - }) - // add this new job to the watcher queue and ensure it succeeds - await addToNextQueue(_queueJobData.initialJobData, { - _nextTradingQueue: WATCHER_Q_NAME, - rawKiteOrderResponse: response, - originalTriggerPrice: triggerPrice - }) - } catch (e) { - console.log( - '[slmWatcher] error adding watcher for new exit market order', - e - ) - } - return Promise.resolve('[slmWatcher] placing exit order') - } catch (e) { - console.log('🔴 [slmWatcher] error. Checker terminated!!', e) - // a promise reject here could be dangerous due to retry logic. - // It could lead to multiple exit orders for the same initial order_id - // hence, resolve - return Promise.resolve('[slmWatcher] error') - } -} - -export default slmWatcher diff --git a/package.json b/package.json index 6c2b5e30..642d789b 100644 --- a/package.json +++ b/package.json @@ -5,7 +5,7 @@ "scripts": { "dev": "NODE_OPTIONS='--inspect' next dev", "build": "next build", - "start": "node ./bootup.js & TZ=Asia/Kolkata next start -H 0.0.0.0 -p ${PORT:-8080}", + "start": "node ./bootup.js & NODE_OPTIONS=--max-old-space-size=1024 TZ=Asia/Kolkata next start -H 0.0.0.0 -p ${PORT:-8080}", "lint": "next lint --quiet", "test": "jest ./__tests__ --testPathIgnorePatterns support/*", "unit-test": "jest ./__tests__/unit --detectOpenHandles", @@ -13,9 +13,7 @@ "format": "prettier-standard --format" }, "lint-staged": { - "*": [ - "prettier-standard --lint" - ] + "*": ["prettier-standard --lint"] }, "dependencies": { "@date-io/date-fns": "^1.3.13", @@ -90,7 +88,5 @@ "pre-commit": "lint-staged" } }, - "cacheDirectories": [ - ".next/cache" - ] + "cacheDirectories": [".next/cache"] } diff --git a/pages/api/order_history.js b/pages/api/order_history.js index 9cfb3581..113a4d47 100644 --- a/pages/api/order_history.js +++ b/pages/api/order_history.js @@ -1,5 +1,5 @@ import withSession from '../../lib/session' -import { syncGetKiteInstance } from '../../lib/utils' +import { getOrderHistory, syncGetKiteInstance } from '../../lib/utils' export default withSession(async (req, res) => { const user = req.session.get('user') @@ -12,8 +12,8 @@ export default withSession(async (req, res) => { const { id: orderId } = req.query - const orderHistory = await kite.getOrderHistory(orderId) - res.json(orderHistory.reverse()) + const orderHistory = await getOrderHistory(kite, orderId) + res.json(orderHistory) }) // 210428200252388 diff --git a/pages/api/trades_day.ts b/pages/api/trades_day.ts index cd9f2149..8f80f3fc 100644 --- a/pages/api/trades_day.ts +++ b/pages/api/trades_day.ts @@ -6,6 +6,7 @@ import { customAlphabet } from 'nanoid' import { tradingQueue, addToNextQueue, TRADING_Q_NAME } from '../../lib/queue' import { ERROR_STRINGS, STRATEGIES_DETAILS } from '../../lib/constants' +import { EXIT_STRATEGIES } from '../../lib/constants' import console from '../../lib/logging' import withSession from '../../lib/session' @@ -66,6 +67,19 @@ async function createJob ({ ) } + if (jobData.exitStrategy === EXIT_STRATEGIES.MULTI_LEG_PREMIUM_THRESHOLD) { + if ( + process.env.DISABLE_COMBINED_PREMIUM && + JSON.parse(process.env.DISABLE_COMBINED_PREMIUM) + ) { + return Promise.reject( + new Error( + 'Combined SL is temporarily unavailable due to memory leak issues..' + ) + ) + } + } + return addToNextQueue( { ...jobData, diff --git a/types/misc.ts b/types/misc.ts index 3636be68..59131f28 100644 --- a/types/misc.ts +++ b/types/misc.ts @@ -31,3 +31,8 @@ export interface DBMeta { _createdOn?: string _updatedOn?: string } + +export enum ASO_TYPE { + OPEN_POSITION_SQUARE_OFF = 'OPEN_POSITION_SQUARE_OFF', + SLL_TO_MARKET = 'SLL_TO_MARKET' +} diff --git a/types/plans.ts b/types/plans.ts index 440448f2..5e06ec19 100644 --- a/types/plans.ts +++ b/types/plans.ts @@ -21,7 +21,7 @@ export interface SavedPlanMeta extends COMMON_TRADE_PROPS { // _collection?: DailyPlansDayKey isAutoSquareOffEnabled: boolean runNow?: boolean - autoSquareOffProps?: { time: string; deletePendingOrders: boolean } + autoSquareOffProps?: { time: string } runAt?: string squareOffTime: string | undefined expiresAt?: string diff --git a/types/trade.ts b/types/trade.ts index b9b1392f..52ecdd04 100644 --- a/types/trade.ts +++ b/types/trade.ts @@ -12,7 +12,7 @@ export interface TradeMeta extends DBMeta { runNow?: boolean runAt?: string squareOffTime: string | undefined - autoSquareOffProps?: { time: string; deletePendingOrders: boolean } + autoSquareOffProps?: { time: string } expiresAt?: string _kite?: unknown // this is only used in jest for unit tests user?: SignalXUser // this is only available once job has been created on server