US Equity
Corporate Actions
Splits
When a company does a stock split, the number of shares each shareholder owns increases and the price of each share decreases. When a company does a reverse stock split, the number of shares each shareholder owns decreases and the price of each share increases. A company may perform a stock split or a reverse stock split to adjust the price of their stock so that more investors trade it and the liquidity increases.
When a stock split or reverse stock split occurs for an Equity in your algorithm, LEAN sends a Split
object to the OnData
on_data
method.
You receive Split
objects when a split is in the near future and when it occurs.
To know if the split occurs in the near future or now, check the Type
type
property.
If you backtest without the Raw
RAW
data normalization mode, the splits are factored into the price and volume. If you backtest with the Raw
RAW
data normalization mode or trade live, when a split occurs, LEAN automatically adjusts your positions based on the SplitFactor
split_factor
. If the post-split quantity isn't a valid lot size, LEAN credits the remaining value to your cashbook in your account currency. If you have indicators in your algorithm, reset and warm-up your indicators with ScaledRaw data when splits occur so that the data in your indicators account for the price adjustments that the splits cause.
To get the Split
objects, index the Splits
splits
object with the security Symbol
symbol
. The Splits
splits
object may not contain data for your Symbol
. To avoid issues, check if the Splits
splits
object contains data for your security before you index it with the security Symbol
.
public override void OnData(Slice slice) { if (slice.Splits.ContainsKey(_symbol)) { var split = slice.Splits[_symbol]; } } public override void OnSplits(Splits splits) { if (splits.ContainsKey(_symbol)) { var split = splits[_symbol]; } }
def on_data(self, slice: Slice) -> None: split = slice.splits.get(self._symbol) if split: pass def on_splits(self, splits: Splits) -> None: split = splits.get(self._symbol) if split: pass
You can also iterate through the Splits
splits
dictionary. The keys of the dictionary are the Symbol
objects and the values are the Split
objects.
public override void OnData(Slice slice) { foreach (var kvp in slice.Splits) { var symbol = kvp.Key; var split = kvp.Value; } } public override void OnSplits(Splits splits) { foreach (var kvp in splits) { var symbol = kvp.Key; var split = kvp.Value; } }
def on_data(self, slice: Slice) -> None: for symbol, split in slice.splits.items(): pass def on_splits(self, splits: Splits) -> None: for symbol, split in splits.items(): pass
To get historical splits, make a history request with the Split
type.
var splits = History<Split>(symbol, TimeSpan.FromDays(4*365));
splits_df = self.history(Split, symbol, timedelta(4*365)) splits = self.history[Split](symbol, timedelta(4*365))
LEAN stores the data for stock splits in factor files. To view some example factor files, see the LEAN GitHub repository. In backtests, your algorithm receives Split
objects at midnight. In live trading, your algorithm receives Split
objects when the factor files are ready.
If you hold an Option contract for an underlying Equity when a split occurs, LEAN closes your Option contract position.
If a split event occurs before your order is filled, the unfilled portion of the order is adjusted automatically, where its quantity is multiplied by the split factor and the limit/stop/trigger price (if any) is divided by the split factor.
Split
objects have the following properties:
Dividends
A dividend is a payment that a company gives to shareholders to distribute profits.
When a dividend payment occurs for an Equity in your algorithm, LEAN sends a Dividend
object to the OnData
on_data
method.
If you backtest with the Adjusted
ADJUSTED
or TotalReturn
TOTAL_RETURN
data normalization mode, the dividends are factored into the price. If you backtest with the other data normalization modes or trade live, when a dividend payment occurs, LEAN automatically adds the payment amount to your cashbook. If you have indicators in your algorithm, reset and warm-up your indicators with ScaledRaw data when dividend payments occur so that the data in your indicators account for the price adjustments that the dividend causes.
To get the Dividend
objects, index the Dividends
dividends
object with the security Symbol
symbol
. The Dividends
dividends
object may not contain data for your Symbol
symbol
. To avoid issues, check if the Dividends
dividends
object contains data for your security before you index it with the security Symbol
symbol
.
public override void OnData(Slice slice) { if (slice.Dividends.ContainsKey(_symbol)) { var dividend = slice.Dividends[_symbol]; } } public override void OnDividends(Dividends dividends) { if (dividends.ContainsKey(_symbol)) { var dividend = dividends[_symbol]; } }
def on_data(self, slice: Slice) -> None: dividend = slice.dividends.get(self._symbol) if dividend: pass def on_dividends(self, dividends: Dividends) -> None: dividend = dividends.get(self._symbol) if dividend: pass
You can also iterate through the Dividends
dividends
dictionary. The keys of the dictionary are the Symbol
objects and the values are the Dividend
objects.
public override void OnData(Slice slice) { foreach (var kvp in slice.Dividends) { var symbol = kvp.Key; var dividend = kvp.Value; } } public override void OnDividends(Dividends dividends) { foreach (var kvp in dividends) { var symbol = kvp.Key; var dividend = kvp.Value; } }
def on_data(self, slice: Slice) -> None: for symbol, dividend in slice.dividends.items(): pass def on_dividends(self, dividends: Dividends) -> None: for symbol, dividend in dividends.items(): pass
To get historical dividends, make a history request with the Dividend
type.
var dividends = History<Dividend>(symbol, TimeSpan.FromDays(4*365));
dividends_df = self.history(Dividend, symbol, timedelta(4*365)) dividends = self.history[Split](Dividend, timedelta(4*365))
For a full example, see the DividendAlgorithmDividendAlgorithm in the LEAN GitHub repository.
Dividend
objects have the following properties:
Symbol Changes
The benefit of the Symbol
class is that it always maps to the same security, regardless of their trading ticker.
When a company changes its trading ticker, LEAN sends a SymbolChangedEvent
to the OnData
on_data
method.
To get the SymbolChangedEvent
objects, index the SymbolChangedEvents
symbol_changed_events
object with the security Symbol
symbol
. The SymbolChangedEvents
symbol_changed_events
object may not contain data for your Symbol
symbol
. To avoid issues, check if the SymbolChangedEvents
symbol_changed_events
object contains data for your security before you index it with the security Symbol
symbol
.
public override void OnData(Slice slice) { if (slice.SymbolChangedEvents.ContainsKey(_symbol)) { var symbolChangedEvent = slice.SymbolChangedEvents[_symbol]; } } public override void OnSymbolChangedEvents(SymbolChangedEvents symbolChangedEvents) { if (symbolChangedEvents.ContainsKey(_symbol)) { var symbolChangedEvent = symbolChangedEvents[_symbol]; } }
def on_data(self, slice: Slice) -> None: symbol_changed_event = slice.symbol_changed_events.get(self._symbol) if symbol_changed_event: pass def on_symbol_changed_events(self, symbol_changed_events: SymbolChangedEvents) -> None: symbol_changed_event = symbol_changed_events.get(self._symbol) if symbol_changed_event: pass
You can also iterate through the SymbolChangedEvents
symbol_changed_events
dictionary. The keys of the dictionary are the Symbol
objects and the values are the SymbolChangedEvent
objects.
public override void OnData(Slice slice) { foreach (var kvp in slice.SymbolChangedEvents) { var symbol = kvp.Key; var symbolChangedEvent = kvp.Value; } } public override void OnSymbolChangedEvents(SymbolChangedEvents symbolChangedEvents) { foreach (var kvp in symbolChangedEvents) { var symbol = kvp.Key; var symbolChangedEvent = kvp.Value; } }
def on_data(self, slice: Slice) -> None: for symbol, symbol_changed_event in slice.symbol_changed_events.items(): pass def on_symbol_changed_events(self, symbol_changed_events: SymbolChangedEvents) -> None: for symbol, symbol_changed_event in symbol_changed_events.items(): pass
If you have an open order for a security when they change their ticker, LEAN cancels your order. To keep your order, in the OnOrderEvent
on_order_event
method, get the quantity and Symbol
of the cancelled order and submit a new order.
public override void OnOrderEvent(OrderEvent orderEvent) { if (orderEvent.Status == OrderStatus.Canceled) { var ticket = Transactions.GetOrderTicket(orderEvent.OrderId); if (ticket.Tag.Contains("symbol changed event")) { Transactions.AddOrder(ticket.SubmitRequest); } } }
def on_order_event(self, order_event: OrderEvent) -> None: if order_event.status == OrderStatus.CANCELED: ticket = self.transactions.get_order_ticket(order_event.order_id) if "symbol changed event" in ticket.tag: self.transactions.add_order(ticket.submit_request)
LEAN stores the data for ticker changes in map files. To view some example map files, see the LEAN GitHub repository.
SymbolChangedEvent
objects have the following properties:
Delistings
When a company is delisting from an exchange, LEAN sends a Delisting
object to the OnData
on_data
method.
You receive Delisting
objects when a delisting is in the near future and when it occurs.
To know if the delisting occurs in the near future or now, check the Type
property.
To get the Delisting
objects, index the Delistings
delistings
object with the security Symbol
symbol
. The Delistings
delistings
objects may not contain data for your Symbol
symbol
. To avoid issues, check if the Delistings
delistings
object contains data for your security before you index it with the security Symbol
symbol
.
public override void OnData(Slice slice) { if (slice.Delistings.ContainsKey(_symbol)) { var delisting = slice.Delistings[_symbol]; } } public override void OnDelistings(Delistings delistings) { if (delistings.ContainsKey(_symbol)) { var delisting = delistings[_symbol]; } }
def on_data(self, slice: Slice) -> None: delisting = slice.delistings.get(self._symbol) if delisting: pass def on_delistings(self, delistings: Delistings) -> None: delisting = delistings.get(self._symbol) if delisting: pass
You can also iterate through the Delistings
delistings
dictionary. The keys of the dictionary are the Symbol
objects and the values are the Delisting
objects.
public override void OnData(Slice slice) { foreach (var kvp in slice.Delistings) { var symbol = kvp.Key; var delisting = kvp.Value; } } public override void OnDelistings(Delistings delistings) { foreach (var kvp in delistings) { var symbol = kvp.Key; var delisting = kvp.Value; } }
def on_data(self, slice: Slice) -> None: for symbol, delisting in slice.Delistings.items(): pass def on_delistings(self, delistings: Delistings) -> None: for symbol, delisting in delistings.items(): pass
The delist warning occurs on the final trading day of the stock to give you time to gracefully exit out of positions before LEAN automatically liquidates them.
if (delisting.Type == DelistingType.Warning) { // Liquidate with MarketOnOpenOrder on delisting warning var quantity = Portfolio[symbol].Quantity; if (quantity != 0) { MarketOnOpenOrder(symbol, -quantity); } }
if delisting.type == DelistingType.WARNING: # Liquidate with MarketOnOpenOrder on delisting warning quantity = self.portfolio[symbol].quantity if quantity != 0: self.market_on_open_order(symbol, -quantity)
The OnSecuritiesChanged
on_securities_changed
event notifies the algorithm when delisted assets are removed from the universe.
To improve iteration speed, LEAN removes delisted securities from the Securities
securities
primary collection. To access all the securities, iterate the Securities.Total
securities.total
property.
// Access all the securities. foreach (var security in Securities.Total) { } // Exclude delisted securities. foreach (var security in Securities.Values) { }
# Access all the securities. for security in self.securities.total: pass # Exclude delisted securities. for security in self.securities.values(): pass
For a full example, see the DelistingEventsAlgorithmDelistingEventsAlgorithm in the LEAN GitHub repository.
Delisting
objects have the following properties:
Examples
Example 1: Split Adjustment On Raw Price
This example demonstrates resetting and warming up an indicator after a split, so the indicator isn't impacted by the price discontinuity.
private dynamic _spy; public override void Initialize() { // Subscribe to raw data _spy = AddEquity("SPY", Resolution.Minute, dataNormalizationMode: DataNormalizationMode.Raw) as dynamic; // Set up EMA indicator with duck typing _spy.Ema = EMA(_spy.Symbol, 50); } public override void OnSplits(Splits splits) { // Check if SPY has split in on-splits handler if (splits.ContainsKey(_spy.Symbol)) { // If so, reset the EMA indicator _spy.Ema.Reset(); // Re-warm up with split adjusted historical data foreach (var bar in History<TradeBar>(_spy.Symbol, 50, Resolution.Minute, dataNormalizationMode: DataNormalizationMode.SplitAdjusted)) { _spy.Ema.Update(bar.EndTime, bar.Close); } } }
def initialize(self) -> None: # Subscribe to raw data self.spy = self.add_equity("SPY", Resolution.MINUTE, data_normalization_mode=DataNormalizationMode.RAW) # Set up EMA indicator with duck typing self.spy.ema = self.ema(self.spy.symbol, 50) def on_splits(self, splits: Splits) -> None: # Check if SPY has split in on-splits handler if self.spy.symbol in splits: # If so, reset the EMA indicator self.spy.ema.reset() # Re-warm up with split adjusted historical data for bar in self.history[TradeBar](self.spy.symbol, 50, Resolution.MINUTE, data_normalization_mode=DataNormalizationMode.SPLIT_ADJUSTED): self.spy.ema.update(bar.end_time, bar.close)
Example 2: Reinvesting Dividend
Reinvesting dividends can significantly enhance long-term investment growth. When dividends are received, instead of cashing them out, investors can use these funds to purchase additional shares of the same stock or mutual fund. This example uses raw price data, and reinvests dividend payments into stock.
private Symbol _spy; public override void Initialize() { // Subscribe to raw data _spy = AddEquity("SPY", Resolution.Minute, dataNormalizationMode: DataNormalizationMode.Raw).Symbol; } public override void OnData(Slice slice) { // Invest if not if (!Portfolio[_spy].Invested) { SetHoldings(_spy, 0.5m); } } public override void OnDividends(Dividends dividends) { // Check if SPY has dividend in on-dividends handler if (dividends.ContainsKey(_spy)) { // Obtain the total dividend payment amount var dividendPerShare = dividends[_spy].Distribution; var totalDividendAmount = dividendPerShare * Portfolio[_spy].Quantity; // Calculate the quantity to reinvest (all dividend) var quantity = Math.Floor(totalDividendAmount / Securities[_spy].Price); // round down // Place order to reinvest MarketOrder(_spy, quantity); } }
def initialize(self) -> None: # Subscribe to raw data self.spy = self.add_equity("SPY", Resolution.MINUTE, data_normalization_mode=DataNormalizationMode.RAW).symbol def on_data(self, slice: Slice) -> None: # Invest if not if not self.portfolio[self.spy].invested: self.set_holdings(self.spy, 0.5) def on_dividends(self, dividends: Dividends) -> None: # Check if SPY has dividend in on-dividends handler if self.spy in dividends: # Obtain the total dividend payment amount dividend_per_share = dividends[self.spy].distribution total_dividend_amount = dividend_per_share * self.portfolio[self.spy].quantity # Calculate the quantity to reinvest (all dividend) quantity = np.floor(total_dividend_amount / self.securities[self.spy].price) # round down # Place order to reinvest self.market_order(self.spy, quantity)