class Iro::Strategy include Mongoid::Document include Mongoid::Timestamps include Mongoid::Paranoia store_in collection: 'iro_strategies' field :slug validates :slug, presence: true, uniqueness: true LONG = 'is_long' SHORT = 'is_short' field :long_or_short, type: :string validates :long_or_short, presence: true field :description has_many :positions, class_name: 'Iro::Position', inverse_of: :strategy belongs_to :stock, class_name: 'Iro::Stock', inverse_of: :strategies KIND_COVERED_CALL = 'covered_call' KIND_LONG_DEBIT_CALL_SPREAD = 'long_debit_call_spread' KIND_SHORT_DEBIT_PUT_SPREAD = 'short_debit_put_spread' KINDS = [ nil, KIND_COVERED_CALL, KIND_LONG_DEBIT_CALL_SPREAD, KIND_SHORT_DEBIT_PUT_SPREAD, ] field :kind def kind_short case kind when KIND_COVERED_CALL 'cc' when KIND_LONG_DEBIT_CALL_SPREAD 'long-spread' when KIND_SHORT_DEBIT_PUT_SPREAD 'short-spread' else '@TODO-zez' end end field :buffer_above_water, type: :float field :threshold_delta, type: :float field :threshold_netp, type: :float field :next_inner_delta, type: :float field :next_outer_delta, type: :float field :next_inner_strike, type: :float field :next_outer_strike, type: :float field :next_spread_amount, type: :float # e.g. $20 for a $2000 NVDA spread def self.for_ticker ticker where( ticker: ticker ) end def breakeven_covered_call p p.inner_strike + p.begin_inner_price end def breakeven_long_debit_call_spread p p.inner_strike - p.max_gain # p.begin_outer_price + p.begin_inner_price end alias_method :breakeven_short_debit_put_spread, :breakeven_long_debit_call_spread def max_gain_covered_call p # return p.begin_inner_price p.begin_inner_price * 100 - 0.66 # @TODO: is this *100 really? end def max_gain_long_debit_call_spread p ## 100 * disalloed for gameui ( p.inner_strike - p.outer_strike - p.begin_outer_price + p.begin_inner_price ) # - 2*0.66 end def max_gain_short_debit_put_spread p ## 100 * disalloed for gameui ( p.outer_strike - p.inner_strike - p.begin_outer_price + p.begin_inner_price ) # - 2*0.66 end def net_amount_covered_call p ( p.begin_inner_price - p.end_inner_price ) end def net_amount_long_debit_call_spread p outer = p.end_outer_price - p.begin_outer_price inner = p.begin_inner_price - p.end_inner_price out = ( outer + inner ) end alias_method :net_amount_short_debit_put_spread, :net_amount_long_debit_call_spread def max_loss_covered_call p p.begin_inner_price*10 # just suppose 10,000% end def max_loss_long_debit_call_spread p out = p.outer_strike - p.inner_strike end def max_loss_short_debit_put_spread p # different out = p.inner_strike - p.outer_strike end def begin_delta_covered_call p p.begin_inner_delta end def begin_delta_long_debit_call_spread p p.begin_outer_delta - p.begin_inner_delta end alias_method :begin_delta_short_debit_put_spread, :begin_delta_long_debit_call_spread def end_delta_covered_call p p.end_inner_delta end def end_delta_long_debit_call_spread p p.end_outer_delta - p.end_inner_delta end alias_method :end_delta_short_debit_put_spread, :end_delta_long_debit_call_spread ## ## decisions ## def calc_rollp_covered_call p if ( p.expires_on.to_date - Time.now.to_date ).to_i < 1 return [ 0.99, '0 DTE, must exit' ] end if ( stock.last - buffer_above_water ) < p.inner_strike return [ 0.98, "Last #{'%.2f' % stock.last} is " + "#{'%.2f' % [p.inner_strike + buffer_above_water - stock.last]} " + "below #{'%.2f' % [p.inner_strike + buffer_above_water]} water" ] end if p.end_inner_delta < threshold_delta return [ 0.61, "Delta #{p.end_inner_delta} is lower than #{threshold_delta} threshold." ] end if 1 - p.end_inner_price/p.begin_inner_price > threshold_netp return [ 0.51, "made enough #{'%.0f' % [(1 - p.end_inner_price/p.begin_inner_price )*100]}% profit" ] end return [ 0.33, '-' ] end ## @TODO def calc_rollp_long_debit_call_spread p if ( p.expires_on.to_date - Time.now.to_date ).to_i < 1 return [ 0.99, '0 DTE, must exit' ] end if ( p.expires_on.to_date - Time.now.to_date ).to_i < 2 return [ 0.99, '1 DTE, must exit' ] end if ( stock.last - buffer_above_water ) < p.inner_strike return [ 0.95, "Last #{'%.2f' % stock.last} is " + "#{'%.2f' % [stock.last - p.inner_strike - buffer_above_water]} " + "below #{'%.2f' % [p.inner_strike + buffer_above_water]} water" ] end if p.end_inner_delta < threshold_delta return [ 0.79, "Delta #{p.end_inner_delta} is lower than #{threshold_delta} threshold." ] end if 1 - p.end_inner_price/p.begin_inner_price > threshold_netp return [ 0.51, "made enough #{'%.0f' % [(1 - p.end_inner_price/p.begin_inner_price )*100]}% profit" ] end return [ 0.33, '-' ] end ## @TODO def calc_rollp_short_debit_put_spread p if ( p.expires_on.to_date - Time.now.to_date ).to_i < 1 return [ 0.99, '0 DTE, must exit' ] end if ( stock.last - buffer_above_water ) < p.inner_strike return [ 0.98, "Last #{'%.2f' % stock.last} is " + "#{'%.2f' % [stock.last - p.inner_strike - buffer_above_water]} " + "above #{'%.2f' % [p.inner_strike + buffer_above_water]} water" ] end if p.end_inner_delta < threshold_delta return [ 0.79, "Delta #{p.end_inner_delta} is lower than #{threshold_delta} threshold." ] end if 1 - p.end_inner_price/p.begin_inner_price > threshold_netp return [ 0.51, "made enough #{'%.0f' % [(1 - p.end_inner_price/p.begin_inner_price )*100]}% profit" ] end return [ 0.33, '-' ] end def to_s "#{stock} #{kind_short} #{slug}" end def self.list long_or_short = nil these = long_or_short ? where( long_or_short: long_or_short ) : all [[nil,nil]] + these.map { |ttt| [ ttt, ttt.id ] } end end