A Quantstrat to Build on Part 2

[This article was first published on Timely Portfolio, and kindly contributed to R-bloggers]. (You can report issue about the content on this page here)
Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.

As I explore additional functionality of quantstrat and make changes to my original post A Quantstrat to Build On, I will write multiple posts, and hopefully, the finished product will not be so overwhelming to comprehend.  Also, it might highlight how I build a system, or how like Pixar, I go “from suck to not suck,” highlighted in a great Fast Company article.  I’ll make 5 changes here:

  1. Add another index GDAXI (German Dax) for additional testing, and I’ll save others for out of sample testing
  2. Change the number of periods (in this case weeks) to 50 from 20 so we have less signals and enjoy life more.  I also added this as a variable early, so we only have to change once to apply across the system and charts
  3. Change the gt(>) –0.5 that I hacked originally to the more correct gte (>=) 0 relationship provided by quantstrat
  4. Added the CUD indicator to the chart.Posn with add_TA (note add_TA works and TA=”addTA..” does not) and removed the log=TRUE option which does not work

I learned that there are 5 (as of 6/7/2011) signal types that can be implemented:

sigComparison is the generic comparison of up to two columns which returns TRUE or FALSE in each period a condition (>=,>,=,<,or<=) is met.  With some hacking this could be used in our scenario if we specified a column with all zeros as our comparison, but this is not recommended and only mentioned to stimulate additional learning (hopefully not confusion).

sigCrossover is a limited sigComparison that only returns TRUE in the period in which a crossover occurs between two columns and NA for all other periods.  This would be used in a moving average type system.  With some hacking this could be used in our scenario if we specified a column with all zeros as our comparison, but this is not recommended and only mentioned to stimulate additional learning (hopefully not confusion).

sigPeak returns TRUE when a single column experiences a local peak or bottom.  This is hard coded to only allow for an analysis of the lag 2 (2 periods ago), lag 1 (1 period ago), and current period, so for instance if a stock closes at 50, 49, and then 49.5, TRUE will be returned.

sigThreshold which is what I use in this example where TRUE is returned if a column moves >=,>,=,<, or <= a specified threshold.  An argument is offered called cross to specify whether TRUE (cross=TRUE) is returned only on the period in which the condition is first matched or TRUE (cross=FALSE) is returned in every period in which the condition is met.  In this example, I use cross=TRUE which works much efficiently later in the addRule portion for in/out long-only type systems.

sigFormula which could be adapted to almost anything.  In the source they mention Vijay Vaidyanthan, past CEO and founder of Return Metrics now at EDHEC, who gave some good presentations at http://www.siliconvalleyaaii.org/archives/archives20072008.html.  I hope to provide some examples of this later, since none of the package demos show this.

From TimelyPortfolio
From TimelyPortfolio

R code:

#thanks so much to the developers of quantstrat
#99% of this code comes from the demos in the quantstrat package   
#in this I make 4 changes to that posted 6/2/2011
#1.Add another index GDAXI (German Dax) for additional testing;
# for now use USD but see faberMC demo for currency fun;
# I'll save more for out of sample testing
#2.Change the number of periods (in this case weeks) to 50 from 20
# so we have less signals and enjoy life more.  I also added this as
# variable early in the code, so we only have to change once to apply
# across the system and charts
#3.Change the gt(>) –0.5 that I hacked originally to the more correct
# gte (>=) 0 relationship provided by quantstrat
#4.Added the CUD indicator to the chart.Posn with add_TA
# (note add_TA works and TA=”addTA..” does not) and removed
# the log=TRUE option which does not work   
#now let's define our silly countupdown function
CUD <- function(price,n) {
	#CUD takes the n-period sum of 1 (up days) and -1 (down days)
	temp<-runSum(ifelse(ROC(price,1,type="discrete") > 0,1,-1),n)
	colnames(temp) <- "CUD"
	temp
}   
try(rm("order_book.CUD",pos=.strategy),silent=TRUE)
try(rm("account.CUD","portfolio.CUD",pos=.blotter),silent=TRUE)
try(rm("port.st","symbols","symbol","stratCUD","initDate","initEq",
	'start_t','end_t','num_periods'),silent=TRUE)   
#specify this for the rolling periods to use for our signal
num_periods=50   
# Initialize a strategy object
stratCUD <- strategy("CUD")  
# Add an indicator
stratCUD <- add.indicator(strategy = stratCUD, name = "CUD", 
	arguments = list(price = quote(Cl(mktdata)),n=num_periods),
	label="CUD")   # enter when CUD > 0
stratCUD <- add.signal(strategy = stratCUD, name="sigThreshold",
	arguments = list(threshold=0, column="CUD",relationship="gte", cross=TRUE),
	label="CUD.gte.0")
# exit when CUD < 0
stratCUD <- add.signal(strategy = stratCUD, name="sigThreshold",
	arguments = list(threshold=0, column="CUD",relationship="lt",cross=TRUE),
	label="CUD.lt.0")   stratCUD <- add.rule(strategy = stratCUD, name='ruleSignal', 
	arguments = list(sigcol="CUD.gte.0", sigval=TRUE, orderqty=1000, ordertype='market',
	 orderside='long', pricemethod='market', replace=FALSE), type='enter', path.dep=TRUE)
stratCUD <- add.rule(strategy = stratCUD, name='ruleSignal', 
	arguments = list(sigcol="CUD.lt.0", sigval=TRUE, orderqty='all',
	 ordertype='market', orderside='long', pricemethod='market', replace=FALSE),
	 type='exit', path.dep=TRUE)   currency("USD")
symbols = c("GSPC","GDAXI")
for (symbol in symbols) {
	stock(symbol, currency="USD",multiplier=1)
	#use paste with ^ to get index data
	getSymbols(paste("^",symbol,sep=""),adjust=T,from="1900-12-31")
	assign(symbol,to.weekly(get(symbol)))
}   
initDate='1950-12-31'
initEq=100000
port.st<-'CUD' 
#use a string here for easier changing of parameters and re-trying   
initPortf(port.st, symbols=symbols, initDate=initDate)
initAcct(port.st, portfolios=port.st, initDate=initDate)
initOrders(portfolio=port.st, initDate=initDate)   
print("setup completed")   
# Process the indicators and generate trades
start_t<-Sys.time()
out<-try(applyStrategy(strategy=stratCUD , portfolios=port.st ) )
end_t<-Sys.time()
print("Strategy Loop:")
print(end_t-start_t)   start_t<-Sys.time()
updatePortf(Portfolio=port.st,Dates=paste('::',as.Date(Sys.time()),sep=''))
end_t<-Sys.time()
print("trade blotter portfolio update:")
print(end_t-start_t)   
# hack for new quantmod graphics, remove later
themelist<-chart_theme()
themelist$col$up.col<-'lightgreen'
themelist$col$dn.col<-'pink'   
for(symbol in symbols){
	dev.new()
	chart.Posn(Portfolio=port.st,Symbol=symbol,theme=themelist)
	#add the CUD indicator to the bottom of the chart
	plot(add_TA(CUD(get(symbol)[,4],n=num_periods)))
}

To leave a comment for the author, please follow the link and comment on their blog: Timely Portfolio.

R-bloggers.com offers daily e-mail updates about R news and tutorials about learning R and many other topics. Click here if you're looking to post or find an R/data-science job.
Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.

Never miss an update!
Subscribe to R-bloggers to receive
e-mails with the latest R posts.
(You will not see this message again.)

Click here to close (This popup will not appear again)