CFA Level 3

Chapter: Capital Market Expectations (5-10%)

Trend Rate of Growth

%Growth=%labor input+%capital per worker+%total factor productivity\%\text{Growth}= \%\text{labor input} + \%\text{capital per worker} + \%\text{total factor productivity}

where:

  • Labor input: labor force (population, demographics) + labor participation (real wages, work/leisure decision, social factors)
  • capital per worker: increases labor productivity
  • total factor productivity: technological progress + changes in government policies

Business Cycle Phases

The business cycle can be subdivided into five phases:

1. Initial Recovery

  • Duration of a few months.
  • Business confidence rising.
  • Government stimulus provided by low interest rates and/or budget deficits.
  • Decelerating inflation.
  • Large output gap.
  • Low or falling short-term interest rates.
  • Bond yields bottoming out.
  • Rising stock prices.
  • Cyclical, riskier assets such as small-cap stocks and high yield bonds doing well.

2. Early Expansion

  • Duration of a year to several years.
  • Increasing growth with low inflation.
  • Increasing confidence.
  • Rising short-term interest rates.
  • Output gap is narrowing.
  • Stable or rising bond yields.
  • Rising stock prices.

3. Late Expansion

  • High confidence and employment.
  • Output gap eliminated and economy at risk of overheating.
  • Increasing inflation.
  • Central bank limits the growth of the money supply.
  • Rising short-term interest rates.
  • Rising bond yields.
  • Rising/peaking stock prices with increased risk and volatility.

4. Slowdown

  • Duration of a few months to a year or longer.
  • Declining confidence.
  • Inflation still rising.
  • Short-term interest rates at a peak.
  • Bond yields peaking and possibly falling, resulting in rising bond prices.
  • Possible inverting yield curve.
  • Falling stock prices.

5. Contraction.

  • Duration of 12 to 18 months.
  • Declining confidence and profits.
  • Increase in unemployment and bankruptcies.
  • Inflation topping out.
  • Falling short-term interest rates.
  • Falling bond yields, rising prices.
  • Stock prices increasing during the latter stages, anticipating the end of the recession.

Taylor rule

central bank uses taylor rule to make monetary policy decisions, determined by the target interest rate using the neutral rate, expected GDP relative to its long-term trend, and expected inflation relative to its targeted amount. It can be formalized as follows:

ntarget=rneutral+iexpected+[0.5(GDPexpectedGDPtrend)+0.5(iexpecteditarget)]adjustmentn_{target} = r_{neutral} + i_{expected} + \underbrace{[0.5(\text{GDP}_{expected} - \text{GDP}_{trend}) + 0.5(i_{expected} - i_{target})]}_{\text{adjustment}}

where:

  • ntargetn_{target} = target nominal short-term interest rate
  • rneutralr_{neutral} = neutral real short-term interest rate
  • GDPexpected\text{GDP}_{expected} = expected GDP growth rate
  • GDPtrend\text{GDP}_{trend} = long-term trend in the GDP growth rate
  • iexpectedi_{expected} = expected inflation rate
  • itargeti_{target} = target inflation rate

Macroeconomic linkage

A country’s current account and capital account are measures of macroeconomic linkages

net exports=net private saving+government surplusXM=(SI)+(TG)\begin{aligned} \text{net exports} &= \text{net private saving} + \text{government surplus} \\ X - M &= (S - I) + (T - G) \end{aligned}

where:

  • X = exports
  • M = imports
  • S = private savings
  • I = investment spending
  • T = tax
  • G = government spending

GK Model

GK Model, i.e. Grinold-Kroner model, states that the expected return of a stock is its dividend yield, plus the inflation rate, plus the real earnings growth rate, minus the change in stock outstanding, plus changes in the P/E ratio:

E(Re)DP+(%ΔE%ΔS)growth rate of eps+%ΔPEDP%ΔSincome (cash flow) return+%ΔEnominal earnings growth+%ΔPErepricing returnDP+gconstant growth model+i%ΔS+%ΔPEDP+g+iNominal GDP Growth(in the long run)\begin{aligned} E(R_e) & \approx \frac{D}{P} + \underbrace{(\%\Delta{E} - \%\Delta{S})}_{\text{growth rate of eps}} + \%\Delta{\frac{P}{E}} \\ & \approx \underbrace{\frac{D}{P} - \%\Delta{S}}_{\text{income (cash flow) return}} + \underbrace{\%\Delta{E}}_{\text{nominal earnings growth}} + \underbrace{\%\Delta{\frac{P}{E}}}_{\text{repricing return}} \\ & \approx \underbrace{\frac{D}{P} + g}_{\text{constant growth model}} + i - \%\Delta{S} + \%\Delta{\frac{P}{E}} \\ & \approx \frac{D}{P} + \underbrace{g + i}_{\text{Nominal GDP Growth}} && \text{(in the long run)} \end{aligned}

where:

  • E(Re)E(R_e) = expected equity return
  • DP\frac{D}{P} = dividend yield
  • %ΔE\%\Delta{E} = expected percentage change in total earnings
  • %ΔS\%\Delta{S} = expected percentage change in shares outstanding, e.g. share repurchase means negative %ΔS\%\Delta{S}, increasing E(Re)E(R_e)
  • %ΔP/E\%\Delta{P/E} = expected percentage change in the P/E ratio
  • all the inputs should be forward-looking (i.e. forecast)

note:

  • a share repurchase is a negative %ΔS\%\Delta{S} term, thus increasing E(Re)E(R_e)
  • a nominal earnings growth rate is used, thus incorporating effects of inflation explicitly
  • limitation: assumption of infinite time horizon, ignoring investment horizon
  • limitation: a %ΔE\%\Delta{E} larger than GDP growth is not plausible, because it implies a perpetually expanding economy

Real Estate Variation:

E(Rre)=cap rate+NOI growth rate%Δcap rateE(R_{re}) = \text{cap rate} + \text{NOI growth rate} - \%\Delta{\text{cap rate}}

ST Model

ST Model, i.e. Singer-Terhaar model, international CAPM

Ri=Rf+βi,M(RMRf)(CAPM)RPi=βi,M×RPM(RP=RRf)=ρi,Mσi(RPMσM)Sharpe Ratio(βi,M=ρi,MσiσM)\begin{aligned} R_i &= R_f + \beta_{i, M} (R_M - R_f) && \text{(CAPM)} \\ RP_i &= \beta_{i, M} \times RP_M && (\because RP = R - R_f) \\ &= \rho_{i, M} \cdot \sigma_i \cdot \underbrace{(\frac{RP_M}{\sigma_M})}_{\text{Sharpe Ratio}} && (\because \beta_{i, M} = \rho_{i, M} \frac{\sigma_i}{\sigma_M}) \end{aligned}

Step1. Full integrated market

RPiG=ρi,GMσi(RPGMσGM)(+ e.g. liquidity premium)\begin{aligned} RP_i^G = \rho_{i, GM} \cdot \sigma_i \cdot (\frac{RP_{GM}}{\sigma_{GM}}) && (\text{+ e.g. liquidity premium}) \end{aligned}

Step2. Full segmented market

RPiS=σi(RPiSσi)(+ e.g. liquidity premium)\begin{aligned} RP_i^S = \sigma_i \cdot (\frac{RP_{i}^S}{\sigma_{i}}) && (\text{+ e.g. liquidity premium}) \end{aligned}

Step3. weighted average risk premium

RP=ϕRPG+(1ϕ)RPSRP = \phi \cdot RP^G + (1 - \phi) \cdot RP^S

where:

  • ϕ\phi = degree of integration with the global markets

Step4. expected return

R=RP+RfR = RP + R_f

Dornbusch overshooting mechanism

immediate capital flows will strengthen the currencies of countries with high expected returns to the point where the high return currency will be expected to depreciate going forward by the return differential.

E(%ΔSd/f)=(rdrf)+(TermdTermf)+(CreditdCreditf)+(EquitydEquityf)+(LiquiddLiquidf)\begin{aligned} E(\%\Delta S_{d/f}) &= (r^d - r^f) \\ &\quad + (\text{Term}^d - \text{Term}^f) \\ &\quad + (\text{Credit}^d - \text{Credit}^f) \\ &\quad + (\text{Equity}^d - \text{Equity}^f) \\ &\quad + (\text{Liquid}^d - \text{Liquid}^f) \end{aligned}

Chapter: Asset Allocation (5-10%)

MVO

MVO, i.e. Mean-variance optimization

max(Utility)=E(R)0.005×λ×Varmax(\text{Utility}) = E(R) - 0.005 \times \lambda \times Var

where:

  • λ\lambda = risk aversion score

note:

  • E® and Var are in % terms, if using decimal, times 0.5 instead of 0.005

limitation:

  • GIGO: results sensitive to inputs, e.g. expected return, standard deviation, and correlation
  • Concentrated asset class allocations: often results in a highly concentrated subset of asset classes, with zero allocation to others, causing lack of diversification of asset classes
  • Skewness and kurtosis: only looks at expected return and variance, while in realty, there is significant skewness and kurtosis in actual returns
  • Liquidity constraint: does not automatically incorporate liquidity constraints, often resulting in over-allocation to illiquid assets.

Black-Litterman Model

Black-Litterman model, starts with the “optimal” portfolio weights from the global market portfolio and derive the expected returns consistent with those weights. Then adjust these return estimates (called implied returns) to do a traditional MVO and derive optimal portfolio weights for our particular investor.

Black-Litterman-Model

e.g.

  • Derives an expected return for emerging market equities of 6.5% and you believe this is too low, you could adjust the expected return by 75 basis points to 7.25%. You can then rerun the MVO using your adjusted return estimates.
  • Projects a return for U.K. large-cap equities of 8.2% and U.S. large-cap equities of 8.0% (a return differential of 20 basis points) and you believe that U.S. large-cap equities will outperform U.K. large-cap equities by 100 basis points, adjust the differential.

pros:

  • world market weights are fully diversified and theoretically optimal
  • less dependent on initial return estimates
  • resulting portfolios are likely to contain more asset classes

Heuristic approach

60/40 rule

60% equity, 40% fixed income for the average individual

120 minus age

120 − age = % allocation to equities, with the remainder going to fixed incomes.

Risk budgeting approach

The goal of risk budgeting is to maximize return per unit of risk (e.g. total portfolio risk, active risk, or residual risk.)

MCTRi=βi,p×σpACTRi=wi×MCTRi\begin{aligned} \text{MCTR}_i &= \beta_{i, p} \times \sigma_p \\ \text{ACTR}_i &= w_i \times \text{MCTR}_i \end{aligned}

where:

  • MCTRi\text{MCTR}_i = Marginal contribution to total risk, i.e. the change in total portfolio risk for a small change in the asset allocation to a specific asset class. i.e. the partial derivative of risk with respect to changes in portfolio allocations
  • βi,p\beta_{i, p} = beta of asset class i with respect to the portfolio
  • σp\sigma_p = total portfolio risk, i.e. standard deviation
  • ACTRi\text{ACTR}_i = Absolute contribution to total risk, % of risk contributed by positioni\text{position}_i= ACTRi\text{ACTR}_i / total portfolio risk

Chapter: Fixed Income

LDI types

Four types of liability-driven investing (i.e. LDI)

  • Type I: Known future amount(s) and payout dates(s)
  • Type II: Known future amount(s) but uncertain payout dates(s)
  • Type III: Uncertain future amount(s) but known payout dates(s)
  • Type IV: Uncertain future amount(s) and uncertain payout dates(s)

Immunization

Steps to compute Portfolio statistics:

  1. Project the time to receipt (starting with the nearest to most distant) of every portfolio cash flow.
  2. Determine the aggregate portfolio cash flow in each period. The analysis uses six-month periods.
  3. Determine the portfolio IRR that equates future cash flows with the current market value of the portfolio.
  4. Use that IRR to determine the PV of each future cash flow from Step 2. (The sum of those PVs will be the current portfolio market value.)
  5. Calculate the PV weight (w) to apply to each payment as its PV (Step 4) divided by the sum of the PVs.
  6. For each cash flow, multiply its (w) by its time until receipt(t). The sum of the (w)(t)s is the portfolio’s Macaulay duration. Duration is normally expressed in years, so if the cash flow periods were in six-month increments, divide by 2 (two six-month periods in a year) for annual duration.
  7. Portfolio dispersion is computed as the weighted average variance of when each cash flow is received around portfolio duration. (Remember, duration is just the weighted average of when all the cash flows are received).
  8. Portfolio convexity can be computed by summing for each cash flow: [(t)(t + 1)(w)] and then divide this sum by (1 + portfolio IRRperiodicIRR_{periodic})2.

note:

  • Portfolio statistics should be used for ALM work rather than traditional weighted average calculations based on each bond. With flat yield curve, there’s no diff. In an upward-sloping yield curve, portfolio duration and cashflow yield (i.e. IRR) will be higher-than-average duration and YTM of the bonds because portfolio statistics reflect all cash flows (and return) to be received and the longer maturity bonds will impact the portfolio for a longer time.
  • The goal of the immunized portfolio is to earn the initial portfolio IRR, not the average YTM of the bonds. Earning the IRR means the portfolio will grow to a sufficient FV to fund the liability.

convexity=Macaulay duration2+Macaulay duration+dispersion(1+periodic IRR)2\text{convexity} = \frac{\text{Macaulay duration}^2 + \text{Macaulay duration} + \text{dispersion}}{(1 + \text{periodic IRR})^2}

The dispersion and convexity will indicate the risk exposure of the immunization strategy to structural risk from shifts and twists in the yield curve.

Cash flow matching strategy

the safest approach, may allow accounting defeasance to legally set aside assets dedicated to meet the liability (both to be removed from the B/S)

bullet, barbell, Laddered portfolios (created by bonds directly or target-date bond ETFs)

Duration matching strategy

more flexible and generally practical approach to funding multiple liabilities.

for single liability:

  • PVA >= PVL
  • MDAMD_A = MDLMD_L
  • minimize Convexity

for multiple liabilities:

  • PVA >= PVL
  • BPVA=BPVLBPV_A = BPV_L
  • CA>=CLC_A >= C_L slightly

Contingent immunization (CI)

a hybrid active/passive strategy. As long as a surplus is significant, the portfolio can be actively managed. if the strategy is unsuccessful, the surplus will shrink, and the portfolio must be immunized before the surplus declines below zero

Hedge Duration Gap between PVA and PVL

Consider an underfunded pension (i.e. DB plan) fund (i.e. PVA < PVL)

1. using future contract

Nf for 100% hedge=BPVLBPVABPV Gapfutures BPVN_f \text{ for 100\% hedge} = \frac{\overbrace{\text{BPV}_L - \text{BPV}_A}^{\text{BPV Gap} }}{\text{futures BPV}}

where:

  • futures BPVBPVCTDCFCTD\text{futures BPV} \approx \frac{\text{BPV}_{\text{CTD}}}{\text{CF}_{\text{CTD}}}

2. using interest rate swaps

receive-fixed swap has positive (+) duration; pay-fixed swap has negative (-) duration

e.g. receive-fixed swap:

net swap BPV=fixed-side BPVfloating-side BPV\text{net swap BPV} = \text{fixed-side BPV} - \text{floating-side BPV}

NP=BPV Gapnet swap BPV\text{NP} = \frac{\text{BPV Gap}}{\text{net swap BPV}}

3. using swaption

  • to increase asset duration (i.e. BPVA\text{BPV}_A, closing duration gap), buy receiver swaption
  • to decrease asset duration (i.e. BPVA\text{BPV}_A, widening duration gap), buy payer swaption

3*. using swaption collar

  • buy a receiver swaption at lower strike (i.e. SFR: swap fixed rate)
  • sell a payer swaption at higher strike (to reduce the initial premium cost outlay)

note:

  • the payoff looks like a short risk reversal, with interest rate as X-axis

Example: choosing an optimal strategy

The choice of optimal strategy will depend on the manager’s view of interest rates. Consider the DB plan with a duration gap and a need to increase asset duration.

  1. Enter a receive-fixed swap versus pay MRR.
  2. Buy a receiver swaption.
  3. Enter a zero-cost collar composed of buying the receiver swaption and selling a payer swaption.
Premium Cost Cost
2.5% fixed-rate swap None
2.3% receiver swaption 75 bp
3.3% payer swaption 75 bp
  • The receive 2.5% SFR swap is optimal if the manager expects the new SFR will be at or below 2.5%.

    • This is equivalent to buying 2.5% fixed-rate bonds (financed by borrowing at MRR), increasing asset duration and BPV. The plan will benefit from the decline in rates.
    • Buying the 2.3% receiver swaption is suboptimal because there is an initial cost, and the 2.3% fixed rate received by the plan is lower.
    • The collar (buy the 2.3% receiver swaption; sell the 3.3% payer swaption) is suboptimal because the 2.3% fixed rate received by the plan is lower. The payer swaption buyer has no rational reason to exercise his right with the new SFR below 3.3%.
  • The collar is optimal if the manager expects the new SFR will be above 2.5% but below 3.3%.

    • The collar (buy the 2.3% receiver swaption and sell the 3.3% payer swaption) has no intrinsic value, which is the best choice.
      • The right to receive 2.3% when rates are above 2.5% has no value.
      • The payer swaption buyer has no rational reason to exercise his right with new SFRs below 3.3%.
    • The other hedges have negative value or zero value with an up-front cost.
      • The swap of receive 2.5% will have negative value when SFRs are above 2.5%.
      • The receiver swaption (right to receive 2.3%) has no value when new SFRs are above 2.5% and required an initial cost.
  • Buying the 2.3% receiver swaption is optimal at some level of new SFRs above 3.3%.

    • The 2.3% receiver swaption has no intrinsic value with new SFRs above 3.3%. But there was an initial premium cost. This is the best case at some level of SFRs above 3.3%.
    • The receive 2.5% swap has increasing negative value as new SFRs increase above 3.3%.
    • The collar also begins to have increasing negative value as new SFRs increase above 3.3%.
      • The receive 2.3% swaption has no value.
      • The 3.3% payer swaption increases in value as SFRs increase above 3.3%, and this is negative value to the seller (the plan).
      • As SFRs increase, that negative value will at some point exceed the initial cost of the receiver swaption, and the receiver swaption would become optimal.
      • The breakeven rate to make the payer swaption optimal is above 3.3%.

Fixed-income return decomposition

    1. rolling yield:
    • a. coupon income = annual coupon amount / current bond price
    • b. rolldown return = (projected bond price (BP) assuming not yield curve change - beginning BP) / beginning BP
    1. expected price change due to change in benchmark yield = (MD×ΔY)+(1/2×C×ΔY2)(-MD \times \Delta{Y}) + (1/2 \times C \times \Delta{Y}^2)
    1. expected price change due to change in credit spreads = (MD×ΔS)+(1/2×C×ΔS2)(-MD \times \Delta{S}) + (1/2 \times C \times \Delta{S}^2)
    1. expected currency G/L

Credit Spread Measures

Yield Spread

yield spread or benchmark spread := bond YTM - closest maturity on-the-run govt bond

g-spread

g-spread := bond YTM - interpolated YTM of the two adjacent maturity on-the-run govt bonds

i-spread

i-spread, interpolated spread := bond YTM - maturity interpolated swap fixed rate

asset swap spread

ASW := bond fixed coupon - maturity interpolated swap fixed rate

represents the spread that the bond is offering over the floating market reference rate (MRR) over its life (assuming the bond is trading close to par). Since the swap can swap fixed rate with MRR, buying the bond and payer swap can synthesize MRR + ASW return.

z-spread

z-spread, zero-volatility spread, takes into account the term structure of spot rates. It uses a trial-and-error calculation to determine a single spread that, when added to risk-free spot rates, discounts the bond’s future cash flows back to its current market value.

note: CDS Spread, where the protection buyer pays a standardized fixed coupon adjusted by an up-front payment/receipt to reflect the fair value of the protection, should in theory, be equal to the z-spread that is earned on the underlying bond, since both are essentially a risk premium paid to the party facing the credit risk of the bond. However, differences between the CDS spread and the z-spread can occur due to technical reasons, such as the underlying bond price trading away from par, accrued interest on the underlying bond, and idiosyncratic features of the CDS protection. The CDS basis:=CDS spreadz-spread\text{CDS basis} := \text{CDS spread} - \text{z-spread}. This is a useful measure for those trading CDS contracts.

option adjusted spread

option adjusted spread, OAS, is the only spread measure appropriate for assessing the credit/liquidity risk of bonds with embedded options. It uses an assumption of interest rate volatility to build an interest rate tree of possible paths for forward risk-free interest rates. Future cash flows are adjusted for the optionality of the bond along each path (i.e., to reflect whether the option would be exercised). The OAS is then a trial-and-error calculation to determine a single spread that, when added to every node of the interest rate tree of risk-free rates, discounts the bond’s adjusted future cash flows back to its current market value. By adjusting the cash flows of the bond to reflect expected interest rate volatility, the uncertainty of the cash flows due to the optionality of the bond has been removed, and the resulting OAS, therefore, does not include any impact of the option on the spread. The OAS measures the spread that is earned for facing the credit and liquidity risk of the issuer—the impact of the option has been removed from the spread.

The OAS is the best measure for consistent comparison of the spreads on bonds with embedded options (e.g., callable, putable) and bonds without embedded options.

Excess Spread

excess spread: spread in excess of the fair spread for suffering credit losses.

Excess Spread:=SpreadEffSpreadDur×ΔSpreadPOD×LGDcredit loss, credit spread\text{Excess Spread} := \text{Spread} - \text{EffSpreadDur} \times \Delta \text{Spread} - \underbrace{\text{POD} \times \text{LGD}}_{\text{credit loss, credit spread}}

Where:

  • EffSpreadDur: Effective Spread Duration
  • POD: Probability of Default
  • LGD: Loss Given Default, i.e. loss severity, 1 - RR (recovery rate)
  • Spread and POD should be de-annualized to match holding period, if given as annualized

note:

  • Credit valuation adjustment (CVA), computed as the present value of sum of expected credit losses := POD * LGD for all periods scaled up by the expected exposure at the time of default. CVA represents the discount that the credit-risky instrument should trade below an equivalent risk-free security in order to compensate investors for their expected credit losses.

butterfly strategy

profit from a view that the curvature of the yield curve is likely to change, by combining long and short positions in bullets (i.e. body) and barbells (wings)

butterfly spread = - (short-term yield) + (2 x medium-term yield) - long-term yield

note:

  • butterfly spread increase == curvature increase == negative butterfly twist == negative butterfly view (frowns more)

Credit Spread in Economic Cycle

Early Expansion (Recovery) Late Expansion Peak Contraction (Recession)
Corporate Defaults Peak Falling Stable Rising
Credit Spread Level Stable Falling Rising Peak
Credit Spread Slope IG Stable Upward sloping Upward sloping Flat
Credit Spread Slope HY Inverted Upward sloping Upward sloping Inverted
Corporate Leverage Falling Stable Rising Peak

credit-spread-economic-cycle

CIRP

covered interest rate parity (i.e. CIRP) states that high interest rate currencies should trade at a forward discount

forward rateD/F=spot rateD/F×(1+rD)T(1+rF)T\text{forward rate}_{D/F} = \text{spot rate}_{D/F} \times \frac{(1 + r_D)^T}{(1+r_F)^T}

where:

  • spot rateD/F\text{spot rate}_{D/F}, forward rateD/F\text{forward rate}_{D/F}: spot and forward rates denoted in direct quotes

Hedge Foreign Coupon-Paying Bonds

Consider an Australian investor who wishes to hedge their exposure to a coupon-paying U.S. Treasury bond. Assume current USD/AUD = 0.8

Position At Initiation Periodic Semiannual Payments for Next 10 Years At the End
U.S. Treasury Bond Pay out USD 100 million to purchase the U.S. bond Receive-fixed USD bond coupon Receive USD 100 million par at maturity of bond
Fixed-Fixed Cross-Currency Swap Receive USD 100 million and pay AUD 125 million in exchange of principal amounts Pay-fixed USD leg Pay USD 100 million and receive AUD 125 million in exchange of principal amounts
Net Flow Pay AUD 125 million principal outflow Receive-fixed AUD payment Receive AUD 125 million principal inflow

the fixed-fixed cross-currency swap can be synthesized by:

Swap Description Interest Paid Interest Received Principal Exchange at Outset Principal Exchange at End
A USD interest rate swap USD Fixed USD Floating None None
B AUD interest rate swap AUD Floating AUD Fixed None None
C Cross-currency basis swap USD Floating AUD Floating Receive in USD principal, pay out AUD principal Pay out USD principal, receive in AUD principal

Unhedged Cross-Currency Carry Trade

the carry trade involves borrowing in a low interest rate currency and depositing in a high interest rate currency, facing risk that high interest rate currency weakens.

total pnl = foreign returns - cost of funds + FX rate (D/F) return

CDS

The protection buyer pays a regular fixed coupon to the protection seller periodically over the life of the contract in return for a payment upon a prespecified credit event on a reference issuer (or issuers). The size of the CDS is specified at the outset and referred to as the notional principal.

cds-mechanics

characteristics:

  • OTC traded credit derivative
  • Bilateral agreement
  • Counterparty credit risk

The fixed coupon paid periodically by the protection buyer is standardized to 1% for investment-grade (IG) issuers and 5% for high-yield (HY) issuers for operational reasons, to make settlement and clearing of contracts more straightforward. Note this standardized coupon is not the fair premium that should be regularly paid by the protection buyer (referred to as the CDS spread) for the credit protection based on credit pricing models.

Scenario Upfront Premium
CDS spread = fixed coupon None
CDS spread > fixed coupon [(CDS spread – fixed coupon) × EffSpreadDurCDS] paid to protection seller
CDS spread < fixed coupon [(fixed coupon – CDS spread) × EffSpreadDurCDS] paid to protection buyer

CDS Price:

CDS Price1+(Fixed CouponCDS Spread)×EffSpreadDurCDS\text{CDS Price} ≈ 1 + (\text{Fixed Coupon} - \text{CDS Spread}) \times \text{EffSpreadDurCDS}

note:

  • This pricing convention conforms with the usual inverse price/yield relationship seen in fixed income because, when CDS spreads rise, the CDS price will fall and vice versa.
  • For higher-quality issuers, the CDS spread will be lower than the fixed coupon and the CDS price will be above par. For lower-quality issuers, the CDS spread will be higher than the fixed coupon and the CDS price will be below par.
  • protection sell’s pnl is computed as [CDS price (t-1) - CDS price (t)] x Notional Principal
Position Profit if CDS Spreads Profit if CDS Prices
Sell protection (long risk) Fall Rise
Buy protection (short risk) Rise Fall

note:

  • buy protection == long CDS spread == short risk == decrease exposure == short bond == short price

CDS long-short strategy

involves buying protection on issuers, where credit spreads are expected to widen relative to other issuers, while simultaneously selling protection on issuers, where credit spreads are expected to narrow relative to other issuers.

CDS curve trades

buying protection at maturities where CDS spreads are expected to rise relative to other maturities and selling protection at maturities where spreads are expected to fall relative to other maturities.

Credit Spread Curve Strategies

profit from a view on the shape or level of the credit spread curve.

Static Credit Spread Curve Strategies

A manager who believes that the current credit spread curve will remain unchanged can earn excess return in the cash market through either:

  • lowering the average credit rating of their bond portfolio;
  • increasing the spread duration by buying and holding longer-dated bonds.

Dynamic Credit Spread Curve Strategies

e.g. expect upward-sloping CDS curve to flatten, buy ST CDS protection, sell LT CDS protection

Economic Stage Typical Curve Feature Cash CDS
Economic recovery HY spreads narrow more than IG spreads Buy HY bonds Sell HY protection
Sell IG bonds Buy IG protection
HY credit curve steepens Buy short-term HY bonds Sell short-term HY protection
Sell long-term HY bonds Buy long-term HY protection
Economic slowdown HY spreads widen more than IG spreads Buy IG bonds Sell IG protection
Sell HY bonds Buy HY protection
HY credit curve flattens/inverts Buy long-term HY bonds Sell long-term HY protection
Sell short-term HY bonds Buy short-term HY protection

Chapter: Equity

HHI

Herfindahl-Hirschman index (i.e. HHI):

HHI=i=1nwi2\text{HHI} = \sum\limits_{i=1}^n{w_i^2}

effective number of stocks=1HHI\text{effective number of stocks} = \frac{1}{\text{HHI}}

where:

  • n = the number of stocks in the portfolio
  • wi{w_i} = the weight of stock i

note:

  • HHI ranges from 1/n (an equally-weighted portfolio) to 1 (a single stock portfolio)
  • mcap-weighted portfolio has number of holdings larger than the number predicted by HHI, due to disproportionate effect of the largest capitalization stocks on the index

Decomposition of realized active return

realized (i.e. ex post) active return can be decomposed into:

RA=(βpkβbk)×Fk+(α+ϵ)R_A = \sum{(\beta_{pk} - \beta_{bk})} \times F_k + (\alpha + \epsilon)

where:

  • βpk\beta_{pk} = the sensitivity of the portfolio to each rewarded factor (k)
  • βbk\beta_{bk} = the sensitivity of the benchmark to each rewarded factor
  • FkF_k = the return of each rewarded factor
  • (α+ϵ\alpha + \epsilon) = the return not explained by exposure to rewarded factors—alpha (α) is the active return attributable to manager skill, and ε is the idiosyncratic return—noise or luck (good or bad) (In practice it is very difficult to distinguish between α and ε)

Active Share

Active Share=12i=1nwp,iwb,i\text{Active Share} = \frac{1}{2}\sum\limits_{i=1}^n{|w_{p,i} - w_{b,i}|}

where:

  • n = total number of securities in the benchmark or the portfolio
  • wp,iw_{p,i} = weight of security i in the portfolio
  • wb,iw_{b,i} = weight of security i in the benchmark

Risk Contribution

The contribution of Asset i to portfolio variance (CViCV_i):

CVi=j=1nwiwjCij=wiCipCV_i = \sum\limits_{j=1}^n{w_i w_j C_{ij}} = w_i C_{ip}

where:

  • wjw_j = Asset j’s weight in the portfolio
  • CijC_{ij} = the covariance of returns between Asset i and Asset j
  • CipC_{ip} = the covariance of returns between Asset i and the portfolio (= j=1nwjCij\sum\limits_{j=1}^n{w_j C_{ij}})

note:

  • The portfolio variance is the sum of each asset’s contribution to portfolio variance plus any unexplained variance

The contribution of Asset i to portfolio active variance (CAViCAV_i):

CAVi=(wpiwbi)RCipCAV_i = (w_{pi} - w_{bi}) RC_{ip}

where:

  • wpiw_{pi} = weight of Asset i in the portfolio
  • wbiw_{bi} = weight of Asset i in the benchmark
  • RCipRC_{ip} = the covariance between the active returns of Asset i and the active returns of the portfolio, which reflects the covariances between the active returns for Asset i and the active returns for each of the n assets in the portfolio (= j=1n(wpjwbj)RCij\sum\limits_{j=1}^n{(w_{pj} - w_{bj}) RC_{ij}})

note:

  • Adding up the CAVs for all the assets in the portfolio will give the variance of the portfolio’s active return (AVpAV_p).

The contribution of Factor i to portfolio variance:

CVi=j=1nβiβjCij=βiCipCV_i = \sum\limits_{j=1}^n{\beta_{i} \beta_{j} C_{ij}} = \beta_i C_{ip}

where:

  • βi\beta_i = sensitivity of portfolio to Factor i (regression coefficient)
  • CijC_{ij} = the covariance of Factor i and Factor j
  • CipC_{ip} = the covariance of Factor i and the portfolio (= j=1nβjCij\sum\limits_{j=1}^n{\beta_{j} C_{ij}})

note:

  • The portfolio variance is the sum of each factor’s contribution to portfolio variance.

Fundamental Law of Active Management

E(RA)=ICBRσRATCE(R_A) = IC \cdot \sqrt{BR} \cdot \sigma_{R_A} \cdot T C

where:

  • E(RA)E(R_A) = expected active return of the portfolio
  • IC = expected information coefficient of the manager, calculated as the correlation between manager forecasts and realized active returns
  • BR = breadth, i.e. the number of truly independent decisions made by the manager each year
  • TC = transfer coefficient, a number between 0 and 1 that measures the level to which the manager is constrained—TC will take a value of 1 if the manager has no constraints, and 0 if the manager is fully constrained
  • σRA\sigma_{R_A} = the manager’s active risk (the volatility of active returns)

Expected Compounded Return

Rg=Raσ2/2R_g = R_a - \sigma^2 / 2

where:

  • RgR_g = geometric compounded returns
  • RaR_a = arithmetic non-compounded returns
  • σ\sigma = portfolio volatility

note:

  • increasing leverage will lower expected geometric compounded returns over time, due to multiplication to σ\sigma is squared, while it’s not to RaR_a.

example:

  • a portfolio falls 2% and then rises by 2%, yielding a compounded return of 0.98 x 1.02 = 0.9996; vs the x10 leveraged portfolio (i.e. falls 20% and then rises by 20%), yielding 0.8 x 1.2 = 0.96

Style-based Active Management Strategy

Value-based approaches: attempt to identify securities that are trading below their estimated intrinsic value, including:

  • Relative value: Comparing price multiples such as P/E and P/B to peers. An undervalued company has an inexplicably low multiple relative to the industry average.
  • Contrarian investing: Purchasing or selling securities against prevailing market sentiment. For instance, buying the securities of depressed cyclical stocks with low or negative earnings.
  • High-quality value: Equal emphasis is placed on both intrinsic value and evidence of financial strength, high quality management, and demonstrated profitability (the “Warren Buffet” approach).
  • Income investing: Focus is on high dividend yields and positive dividend growth rates.
  • Deep-value investing: Focus is on extremely low valuations relative to assets (e.g., low P/B), often due to financial distress.
  • Restructuring and distressed debt investing: Investing prior to or during an expected bankruptcy filing. The goal is to release value through restructuring the distressed company or through the company having sufficient assets in liquidation to generate appropriate returns.
  • Special situations: Identifies mispricings due to corporate events such as divestitures, spin-offs, or mergers.

Growth-based approaches attempt to identify companies with revenues, earnings, or cash flows that are expected to grow faster than their industry or the overall market. Analysts will be less concerned about high valuation multiples and more concerned about the source and persistence of the growth rates of the company, including:

  • Consistent long-term growth.
  • Shorter-term earnings momentum.
  • GARP (growth at a reasonable price); looking for growth at a reasonable valuation. Often this strategy will use the P/E-to-growth (PEG) ratio, which is calculated as the stock’s P/E ratio divided by expected earnings growth in percentage terms.

Portfolio Management Approaches

Investment Style Description Active Share and Active Risk
Pure indexing No active positions: portfolio is equal to the benchmark Zero Active Share and zero active risk
Factor neutral No active factor bets—idiosyncratic risk low if diversified Low active risk—Active Share low if diversified
Factor diversified Balanced exposure to risk factors and minimized idiosyncratic risk through high number of securities in portfolio Reasonably low active risk—high Active Share from large amount of securities used that are unlikely to be in the benchmark
Concentrated factor bets Targeted factor bets—idiosyncratic risk likely to be high High Active Share and high active risk
Concentrated stock picker Targeted individual stock bets Highest Active Share and highest active risk

Portfolio-Management-Approaches

  • A closet indexer is defined as a fund that advertises itself as being actively managed but is substantially similar to an index fund in its exposures
  • A sector rotator would need to have large permitted deviations in sector weights
  • A stock picker would need to have large permitted deviations in individual security weights
  • A diversified multi-factor investor has large sector deviations and limited single-security deviation, and low tracking error

Chapter: Derivative and Currency Management

Option Strategy

Covered Call

Protective Put

Collar

Straddle

Bull Spread and Bear Spread

Calendar Spread

option-theta

Options that are close to ATM have the highest thetas, and these increase as expiration approaches. In other words ATM options lose time value at an increasing rate as they mature.

calendar spreads exploit the difference in theta between close-to-expiry and more-distant-from-expiry options. For options that are near to ATM, the nearer-dated options will have a higher absolute value of theta (more negative) than longer-dated options.

A long calendar spread: buying longer-dated options and selling shorter-dated options with the same strike and underlying. In principle the premium on the shorter-dated should fall faster than the premium on the longer-dated. Thus more value is gained on the short position than is lost on the long position, and a net profit is realized. The options need to be close to ATM, and little movement should be anticipated in the underlying over the period to expiry of the nearer-dated option (large movement might undermine the profit from the strategy). Both options will either be calls or puts, and the choice between calls and puts will reflect the investor’s view on the longer-term prospects for the stock (calls if bullish, puts if bearish).

A short calendar spread: selling longer-dated options and buying shorter-dated options with the same strike and underlying. When options are sufficiently ITM or OTM the thetas are relatively higher for the longer-dated options. the belief is that the longer-dated options will lose time value relatively more rapidly, thus the position as a whole should gain. The short calendar spread strategy is vulnerable, however, to the underlying moving so the options end up ATM when the shorter-dated option expires (so the longer-dated option premiums rise, unless implied volatility also falls, and overall there is a loss since we are short). If the stock moves at all during the period of the strategy, it would be better for it to move a lot.

In general:

  • A long calendar spread will benefit from a stable market or an increase in implied volatility.
  • A short calendar spread will benefit from a big move in the underlying market or a decrease in implied volatility.

Risk reversal

volatility skew, is where implied volatility increases for more OTM puts, and decreases for more OTM calls. This is explained by OTM puts being desirable as insurance against market declines (so their values are bid up by higher demand, and higher values imply higher volatility), while the demand for OTM calls is low.

If a trader believes that put implied volatility is relatively too high, compared to that for calls, a long risk reversal could be created by buying the OTM call (seen as relatively underpriced) and selling the OTM put (seen as relatively overpriced) for the same expiration.

The net positive delta exposure is then hedged by shorting the underlying stock.

BPVHR

BPVHR, i.e. Basis Point Value Hedge Ratio

BPVHR=BPVtargetBPVportBPVfutures=BPVtargetBPVportBPVCTD×CF\begin{aligned} \text{BPVHR} & = \frac{\text{BPV}_{\text{target}} - \text{BPV}_{\text{port}}}{\text{BPV}_{\text{futures}}} \\ & = \frac{\text{BPV}_{\text{target}} - \text{BPV}_{\text{port}}}{\text{BPV}_{\text{CTD}}} \times \text{CF} \end{aligned}

where:

  • BPVCTD\text{BPV}_{\text{CTD}} = MDCTD×0.0001×[price/100×contract size]\text{MD}_{\text{CTD}} \times 0.0001 \times [\text{price} / 100 \times \text{contract size}]
  • BPVfuturesBPVCTDCFCTD\text{BPV}_{\text{futures}} \approx \frac{\text{BPV}_{\text{CTD}}}{\text{CF}_{\text{CTD}}}
  • CF: conversion factor

FOMC % probability of rate change

FOMC, i.e. The Federal Open Market Committee

effective rate implied by futurecurrent fed target ratefed fund rate assuming changecurrent fed target rate\frac{ \text{effective rate implied by future} - \text{current fed target rate} }{ \text{fed fund rate assuming change} - \text{current fed target rate} }

Cross-Currency Basis & Swap

Cross-Currency Basis represents the additional cost of borrowing dollars synthetically with a currency swap relative to the cost of borrowing directly in the USD cash market. Typically, when describing basis we view it from the foreign-currency perspective rather than the USD perspective. If the cost of borrowing dollars synthetically via a swap is greater than the cost of direct USD borrowing, the foreign currency is said to be exhibiting negative basis. Most currencies have shown a negative basis against the dollar since the financial crisis of 2008. The implication is that the USD borrower must accept a lower interest rate on the foreign-currency interest payments it receives.

currency swap only exchanges interest payments. But cross-currency basis swap exchanges both interest and notional principal of each currency at the beginning and end of the swap. Typically, the periodic payments are floating for floating.

Rationale:
A foreign company requiring USD, might not have access to direct USD borrowing or finds it prohibitively expensive.

e.g.:
A French company requires $50 million to invest in US. The cost to borrow USD directly is MRR + 100 bp. it decided to borrow for 4 years in euros at Euribor + 60 bp with interest paid quarterly, and enter a currency swap to exchange euros for dollars. Basis on the Eurodollar swap is being quoted at –20 bps. The swap pays variable interest on both legs on a quarterly settlement basis. The current $/€ exchange rate is $1.1815.

The three-month euro reference rate is 1.5% and U.S. dollar reference rate is 2.0% at swap initiation. Three months later at the first settlement date, the three-month euro reference rate is 1.6% and the U.S. dollar reference rate is 1.9%.

  • At initiation

Principal flows: $50,000,000 / $1.1815 = €42,319,086

the company borrows €42,319,086 and exchange it for $50,000,000. These amounts will be swapped back at maturity.

  • At first settlement

Pays:

  • € interest on the loan: €42,319,086 × (0.015 + 0.006) × 90 / 360 = €222,175
  • $ interest on the swap: $50,000,000 × 0.02 × 90 / 360 = $250,000

Receives:

  • € interest on the swap: €42,319,086 × (0.015 − 0.002) × 90 / 360 = €137,537

Cost of $ financing:

  • Cost of borrowing $ direct: $50,000,000 × (0.02 + 0.01) × 90 / 360 = $375,000 (U.S. dollar reference rate + 100 bp)

  • Cost of synthetic $ borrowing: $50,000,000 × (0.02 + 0.006 + 0.002) × 90 / 360 = $350,000 (U.S. dollar reference rate + 80 bp)

  • Net benefit of swap: $375,000 − $350,000 = $25,000

  • Net benefit of swap: $50,000,000 (1% − 0.8%) × 90 / 360 = $25,000

  • At second settlement

Pays:

  • € interest on the loan: €42,319,086 × (0.016 + 0.006) × 90 / 360 = €232,755
  • $ interest on the swap: $50,000,000 × 0.019 × 90 / 360 = $237,500

Receives:

  • € interest on the swap: €42,319,086 × (0.016 − 0.002) × 90 / 360 = €148,117

Cost of $ financing:

  • Cost of borrowing $ direct: $50,000,000 × (0.019 + 0.01) × 90 / 360 = $362,500 (U.S. dollar reference rate + 100 bp)
  • Cost of synthetic $ borrowing: $50,000,000 × (0.019 + 0.006 + 0.002) × 90 / 360 = $337,500 (U.S. dollar reference rate + 80 bp)
  • Net benefit of swap: $362,500 − $337,500 = $25,000
  • Net benefit of swap: $50,000,000 (1% − 0.8%) × 90 / 360 = $25,000

note:

  • same principal amounts are exchanged at maturity as initialization. (i.e. no exchange risk on principal)
  • the European company is the dollar payer; the swap dealer is the euro payer.
  • By borrowing in euros and entering a currency swap, the company has locked into a cost of the U.S. dollar reference rate + 80 bp for their USD borrowing, reflecting the 60 bp spread above the euro reference rate on the loan and the –20 bp on the swap.

VIX

CBOE Volatility Index, i.e. VIX, measures implied volatility in the S&P 500 Index over a forward period of 30 days. VIX computes a weighted average of implied volatility inferred from S&P 500 traded options (calls and puts) with an average expiration of 30 days. The VIX Index value is the annualized standard deviation of the expected +/− percentage moves in the S&P 500 Index over the following 30 days.

e.g. if the VIX was at 20, we could interpret it as telling us that the market expects that the S&P will stay within a +/−20% range over one year with a 68% level of confidence. This implies a range +/− 2012=5.77%\frac{20}{\sqrt{12}} = 5.77\% over the next 30-day period.

VIX futures

The VIX futures price can be interpreted as the expected S&P 500 Index volatility in the 30-day period after the futures contract expiration date. An equity holding can be protected from extreme downturns (tail risk) by buying VIX futures. Selling VIX futures creates a short volatility position and captures the volatility risk premium embedded in S&P 500 options. Short volatility positions can result in large losses if expected volatility rises significantly. The term structure of VIX futures can provide insights into the market’s expectations of volatility over time.

VIX Options

VIX options are cash-settled European-style options. VIX options can only be exercised at contract maturity; therefore, the value of the option is determined by the expectations of VIX at the contract expiry.

Other volatility Indexes

Volatility indexes also exist on European stock market indexes. Using similar methodology to the VIX Index, VSTOXX is an implied volatility-based index based on the EURO STOXX 50 Index. VFTSE is a volatility index based on the FTSE 100, and VDAX-NEW is based on the DAX 30.

Variance Swaps

The party receiving the variable payment (the purchaser) will gain on the contract when the realized variance is greater than the implied variance and will lose when the realized variance is less than the implied variance. A variance swap can, therefore, be viewed as a pure play on whether realized variance will be higher or lower than expected variance (implied variance) over the tenor of the swap.

variance-swap

There is no exchange of notional principal at the initiation of the swap. A variance swap also has no interim settlement periods. With a variance swap, there is a single payment at the expiration of the swap based on the difference between actual and implied variance over the life of the swap:

settlement amountT=(variance notional)(realized variancevariance strike)\text{settlement amount}_T = (\text{variance notional})(\text{realized variance} - \text{variance strike})

The value of the swap is zero at initiation because implied volatility is the best ex ante estimate of realized volatility.

Realized variance is calculated by taking the natural log of the daily price relatives, the closing price on day t, divided by the closing price on day t − 1:

Ri=ln(Pt/Pt1)R_i = ln(P_t / P_{t-1})

If having N days of traded prices, we can compute N − 1 price relatives ®:

daily variance=i=1N1Ri2(N1)\text{daily variance} = \sum_{i=1}^{N-1} \frac{R_i^2}{(N-1)}

annualized variance = daily variance × 252

The notional amount for a variance swap can be expressed as either variance notional (NvarN_{var}) or vega notional (NvegaN_{vega}).

pnl=Nvar×(σ2K2)=Nvega×(σ2K22K)\begin{aligned} pnl &= N_{\text{var}} \times (\sigma^2 - K^2) \\ &= N_{\text{vega}} \times (\frac{\sigma^2 -K^2}{2K}) \end{aligned}

since:

variance notional=vega notional2×strike price (K)\text{variance notional} = \frac{\text{vega notional}}{2 \times \text{strike price (K)}}

Currency Quotes

The Price and Base Currencies: The base currency is the denominator of the exchange rate and it is priced in terms of the numerator. Unless clearly identified otherwise, the terms “buy” and “sell” refer to the base currency. e.g. sell spot 1,000,000 at CAD/USD 0.9800 is assumed to mean sell for “immediate delivery” 1,000,000 U.S. dollars and buy 980,000 Canadian dollars.

Bid/Asked Rules: Currencies are quoted with a bid/offered or bid/asked price. By convention, the smaller number is written first and the larger number is second. e.g. A quote of 0.9790/0.9810 CAD/USD. The customer pays the bid/ask spread, paying more and/or receiving less in the transaction.

Forward points are an adjustment to the spot price to determine the forward price, e.g.

Spot Quote Forward Points Points with Decimal Adjusted Forward Price
1.33 1.1 1.1 / 100 = 0.011 1.33 + 0.011 = 1.341
2.554 –9.6 –9.6 / 1,000 = –0.0096 2.554 − 0.0096 = 2.5444
0.7654 13.67 13.67 / 10,000 = 0.001367 0.7654 + 0.001367 = 0.766767

Spot and forward bid/asked, e.g.

Maturity/Settlement Spot Quote/Forward Points
Spot AUD/EUR 1.2571/1.2574
30 days –1.0/–0.9
90 days +11.7/+12.0

FX Swap, a misnomer, rolls over a maturing forward contract using a spot transaction into a new forward contract. An existing forward is “swapped” for another forward transaction.

Currency effects on return and risk

return

RDC=(1+RFC)(1+RFX)1RFC+RFX\begin{aligned} R_{DC} &= (1+R_{FC})(1+R_{FX}) - 1 \\ &\approx R_{FC} + R_{FX} \end{aligned}

note:

  • RDCR_{DC} = asset return denoted in domestic currency
  • RFCR_{FC} = asset return denoted in foreign currency
  • RFXR_{FX} = percentage change in value of the foreign currency denoted in D/F (use domestic/foreign and then solve as EV / BV − 1 = RFXR_{FX})

risk

σRDC2σRFC2+σRFX2+2σRFCσRFXρ(RFC,RFX)\sigma_{R_{DC}}^2 \approx \sigma_{R_{FC}}^2 + \sigma_{R_{FX}}^2 + 2\sigma_{R_{FC}}\sigma_{R_{FX}}\rho_{(R_{FC}, R_{FX})}

note:

  • for domestic investor, σRDC2\sigma_{R_{DC}}^2 is usually higher than σRFC2\sigma_{R_{FC}}^2, meaning more risk
  • correlation between RFCR_{FC} and RFXR_{FX} matters:
    • if positive, RFCR_{FC} is amplified by RFXR_{FX}, increasing the volatility to domestic investor
    • if negative, RFCR_{FC} is dampened by RFXR_{FX}, decreasing the volatility to domestic investor

special case if RFCR_{FC} is a risk-free return:

σRDC=σRFX(1+RFC)\sigma{R_{DC}} = \sigma_{R_{FX}} (1 + R_{FC})

minimum variance hedge ratio (MVHR) is a mathematical approach to determining the hedge ratio. When applied to currency hedging, it is a regression of the past changes in value of the portfolio RDCR_{DC} to the past changes in value of the RFXR_{FX} to minimize the variance of RDCR_{DC}. The hedge ratio is the beta (slope coefficient) of that regression. e.g.

RDCR_{DC} = 0.12 + 1.25(%ΔSUSD/EURS_{USD/EUR}) + ε

hedge ratio = 1.25, the manager will short EUR 1.25 times a long EUR exposure in the portfolio

NDFs

NDFs, i.e. Non-Deliverable Forward, alternative to deliverable forwards and require a cash settlement of gains or losses in a developed market currency at settlement rather than a currency exchange. Emerging market governments frequently restrict movement of their currency into or out of the country to settle normal derivative transactions. Such countries have included Brazil (BRL), China (CNY), and Russia (RUB).

A benefit of NDFs is lower credit risk because delivery of the notional amounts of both currencies is not required. Only the gains to one party are paid at settlement, made in the developed market currency.

Chapter: Alternative Investment

Hedge Fund Strategy

six strategy categories:

    1. Equity related
    1. Event driven
    1. Relative value
    1. Opportunistic
    1. Specialist
    1. Multi-manager

Ranked by overall long exposure to market

L/S Equity:

L/S Equity, i.e. Long/Short Equity. The fund manager longs stocks that they think will rise, and shorts stocks that they believe will fall.

  • typically have 40% - 60% net long exposure, benefitting from market’s long term upward trend
  • more concentrated on particular sector or industry
  • aspire to provide returns comparable to long-only fund, but with half of volatility
  • may use leverage to achieve worthwhile returns

Example:

  • Long extension: net exposure of 100%. e.g. 130/30 fund (gross exposure 160%)
EMN:

EMN, i.e. Equity Market Neutral

  • near-zero market risk exposure; (immune to overall market movements)
  • systematic approach to take long and short positions in diversified stocks
  • alpha seeking, low volatility
  • leverage is generally applied to achieve acceptable level of return
Dedicated Short Selling and Short-Biased

dedicated short-selling: pure shorts overpriced stocks (e.g. poorly managed, in declining market segment, or with deceitful accounting); typically 60% - 120% short (by holding the rest in cash).

short-biased, similar except somewhat offset by a long exposure

  • negative corr with market, lower returns, greater volatility, little leverage
  • activist short selling: not only shorts, but also presents research that contends overprice

Event driven

  • soft-catalyst event-driven approach: investment made before an event is being announced
  • hard-catalyst event-driven approach: investment made after a corporate event is being announced, taking advantage of security prices that have not fully adjusted.

note: Soft-catalyst investing is generally more volatile and, thus, riskier than a hard-catalyst approach.

Merger Arbitrage

long target company stock, and short acquiring company stock. Reverse if assume a failed merger (e.g., antitrust regulation)

  • possibly 300% to 500% leverage for returns
  • high sharpe ratios (relatively steady returns), but large left-tail risk (merger unexpectedly fails)
Distressed Securities

take positions in the securities of firms that are in financial distress, including firms that are in bankruptcy or near bankruptcy. Firms may find themselves in this position for a number of reasons, including too much leverage, difficulty competing in their sector, or accounting issues. The securities of such a firm will often trade at greatly depressed prices.

Compounding the discounting of the securities of distressed firms is the fact that institutions such as insurance companies and banks are often not permitted to hold non-investment-grade securities. The selling of such securities can create significant pricing inefficiencies and can open up opportunities for hedge funds seeking profit.

Relative Value

attempt to exploit valuation differences between securities

Fixed-Income Arbitrage

take advantage of temporary mispricing of fixed-income instruments, by going long comparatively undervalued securities, and going short comparatively overvalued securities, under the assumption that prices will revert toward their fair values.

  • Yield curve trades: anticipated changes in yields
  • Carry trades: long high-yielding and short low yielding

note:

  • substantial leverage is often applied, since fixed-income securities tend to be priced fairly efficiently, the amount of profit that can be earned by fixed-income arbitrage is somewhat limited. 400% leverage is not uncommon; even 1500% leverage is not unheard of.
Convertible Bond Arbitrage

One way to view convertible bonds is as a regular bond plus a long call option on the corresponding stock

exploit the fact that the options within convertible instruments usually exhibit low implied volatilities when compared to the historical volatilities of the equities that underlie the option. To do this without taking on excess risk, convertible bond arbitrageurs will take on other positions to try to hedge out the delta and gamma risk of the convertible bond holdings.

buy relatively undervalued convertible bond and short the relatively overvalued underlying stock

  • delta-hedge based on delta of the convertible bond, typically 300% long vs 200% short
  • profitable if realized equity volatility exceeds implied volatility of convertible’s embedded option net of costs.

Opportunistic

top-down focus on regions, sectors, asset classes (not individual securities). Diversification potential with traditional assets, often with beneficial right-tail skew. Highly liquid, and high leverage.

Global Macro

identify global economic changes (e.g. inflation, FX, rates). may use significant leverage (600% - 700%). Can have high alpha and strong diversification potential. Successful manager being contrarian, investing ahead of others.

Managed Futures

invest long-short via derivatives, usually with high leverage (built-in feature of margin trading). Possible crowding. Typically systematic. many based on volatility or momentum. use signals for exit.

Specialist

Volatility trading

exploit mispriced volatility (exploit skew and smile volatility surface), using:

  • options (e.g. straddles, calendar and price spreads)
  • futures on VIX
  • Variance swaps

note:

  • potential large gains due to convexity of volatility derivatives
  • hard to benchmark
Reinsurance / Life settlements
  • Life settlement: HF buys policy from insured, takes over premium payments and receives death benefits

  • Catastrophe risk reinsurance: HF buys earthquake, tornado, hurricane, flood, etc insurance from reinsurer. Considerable expertise required. Typically illiquid.

Multi-manager

A portfolio of various HF diversified strategies

FoF (Funds-of-funds)

typically 2% management fees + 20% performance incentive on gross gains net of management fees & expenses. Plus fees charged by FoF manager. Lack of transparency into individual HFs; principle-agent issues. Multi-layer fees, no performance fee netting across managers. May include HFs otherwise inaccessible.

Multi-strategy funds

One org, better knowledge over when, how much capital and leverage, of fund correlations and risks. Ease of reallocation between strategies. Absorb netting risk internally (favorable fees). Less diversified operational risk.

Investment Opportunity Set

  • traditional approach:

Liquidity-Based-Investment-Opportunity-Set

Economic-Environment-Based-Investment-Opportunity-Set

  • risk factor based approach:

defining asset classes by statistically estimating their sensitivities to risk factors, e.g. economic growth and inflation, interest rates and credit spreads, or currency values, liquidity, capitalization, and value-versus-growth.

advantage:

  • Identifying sources of risk that are common across asset classes, mitigating a limitation of the traditional approaches, which may classify investments into different classes even when they face largely the same risk factors, leading a manager to believe the portfolio is more diversified than it actually is.
  • by allowing a manager to analyze multiple dimensions of portfolio risk, this approach is useful for developing an integrated risk management framework. In this sense, it can be more useful than the traditional approaches for highlighting the primary drivers of portfolio risk.

disadvantage:

  • risk factor estimates can be sensitive to the period used for analysis
  • The results may also be more difficult to communicate to decision makers and to implement compared to traditional approaches.

Chapter: Private Wealth Management

TTLLU-RR

set of consideration for private wealth management

  • T: Time horizon;
  • T: Taxes;
  • L: Liquidity
  • L: Legal issues
  • U: Unique Constraints
  • R: Returns
  • R: Risk
    • ATBR: ability to bear risk
    • WTBR: willingness to bear risk

SAMURAI

criteria for a good benchmark

  • S: Specified in advance
  • A: Appropriate and consistent with the manager’s investment style
  • M: Measurable
  • U: Unambiguous, able to clearly identify the securities
  • R: Reflective of current investment opinions
  • A: Accountable, accepted by the manager
  • I: Investible, possible to replicate passively

Portfolio Reporting

Portfolio reporting enables private clients to understand how their investment portfolio is performing and whether their financial goals are likely to be achieved. It provides a basis for a private wealth manager to review a client’s IPS and investment strategy with the client to determine if changes are required to achieve the client’s goals.

  • Performance summary for the current period.
  • Market commentary for the current period to provide context for the portfolio’s performance.
  • Portfolio asset allocation at the end of the current period, including strategic asset allocation - weights or tactical asset class target ranges.
  • Detailed performance of asset classes and individual securities.
  • Benchmark report comparing asset class and overall portfolio performance to appropriate benchmarks.
  • Historical performance of client’s investment portfolio since inception.
  • Transaction details for the current period (e.g., contributions, withdrawals, interest, dividends, and capital appreciation).
  • Purchase and sale report for the current period.
  • Impact of currency exposure and exchange-rate fluctuations.
  • Progress toward meeting goal portfolios when using a goals-based investing approach.

Portfolio Review

A portfolio review enables the private wealth manager to reassess a client’s IPS and investment strategy in light of recent performance to determine if changes are required. A portfolio review typically addresses the following areas:

  • Appropriateness of client’s existing goals and investment parameters and if any changes are required.
  • Rebalancing of portfolio asset allocation to target allocation or ranges.
  • Any changes to the wealth manager’s ongoing management of the portfolio (e.g., degree of - discretionary authority).
  • Any changes or updates in the wealth manager’s duties and responsibilities.
  • Any changes to IPS and portfolio review frequency.

Balance Sheet Management of Banks and Insurers

how changes in the market value of assets, liabilities, and leverage levels affect the change in the market value of equity is:

%ΔE=%ΔAM%ΔL(M1)\%\Delta E = \%\Delta A \cdot M - \%\Delta L \cdot (M - 1)

where:

  • %ΔE\%\Delta E = percentage change in the value of equity
  • %ΔA\%\Delta A = percentage change in the value of assets
  • %ΔL\%\Delta L = percentage change in the value of liabilities
  • M = leverage multiplier, A / E

The sensitivity of the institution’s equity capital to a unit change in the reference yield, y, of the assets (i.e., the modified duration of the equity capital) is:

DE=DAMDL(M1)ΔiΔyD_E = D_A \cdot M - D_L \cdot (M - 1) \cdot \frac{\Delta i}{\Delta y}

where:

  • DED_E = modified duration of the institution’s equity capital
  • DAD_A = modified duration of the institution’s assets
  • DLD_L = modified duration of the institution’s liabilities
  • M = leverage multiplier, A / E
  • ΔiΔy\frac{\Delta i}{\Delta y} = estimated change in yield of liabilities, i, relative to a unit change in yield of assets, y

Expected volatility of the percentage changes in the market value of equity capital:

σE2=M2σA2+(M1)2σL22M(M1)σAσLρAL\sigma_E^2 = M^2 \sigma_A^2 + (M - 1)^2 \sigma_L^2 - 2 M (M - 1) \sigma_A \sigma_L \rho_{AL}

where:

  • σE\sigma_E = standard deviation of percentage change in the market value of equity
  • σA\sigma_A = standard deviation of percentage change in the value of assets
  • σL\sigma_L = standard deviation of percentage change in the value of liabilities
  • M = leverage multiplier, A / E
  • ρAL\rho_{AL} = correlation of percentage value changes in assets and liabilities

Estate Planing

Lifetime Gifts vs Testamentary Bequests:

FVgift=[1+rg(1tg)]n(1Tg)FVbequest=[1+re(1te)]n(1Te)RV=FVgiftFVbequest\begin{aligned} & FV_{\text{gift}} &&= [1 + r_g(1-t_g)]^n(1-T_g) \\ & FV_{\text{bequest}} &&= [1 + r_e(1-t_e)]^n(1-T_e) \\ & RV &&= \frac{FV_{\text{gift}}}{FV_{\text{bequest}}} \end{aligned}

note:

  • RV of tax-free gift, Tg=0T_g = 0

Life Insurance cost comparison

  • Net payment cost index: assumes the individual dies at the end of the horizon and cash value is not considered
  • Net surrender cost index: assumes the individual terminates the policy (insurance ceases) at the end of the horizon and the cash value is received

steps:

    1. use annuity due (BGN mode) to compute FV of the premium
    1. use ordinary annuity (END mode) to compute FV of the dividend
    1. FVpremiumFVdividendFV_{\text{premium}} - FV_{\text{dividend}}, further minus cash value if using new surrender cost index
    1. use annuity due (BGN mode) to compute annuitized cost

Growth-adjusted discount rate

r=(1+r)/(1+g)1r = (1 + r) / (1 + g) - 1

Taxation

After-Tax Holding Period Return

R=value1value0+incometaxvalue0R' = \frac{value_1 - value_0 + income - tax}{value_0}

or

R=Rtaxvalue0R' = R - \frac{tax}{value_0}

note:

  • if any intermediate cashflow paid as income, adjust value0value_0 using the way used in modified dietz return

After-Tax Post-Liquidation Return

RPL=(1+R1)(1+R2)(1+Rn)(1liquidity tax/final value)n1R_{PL} = \sqrt[n]{(1 + R'_1)(1 + R'_2) \cdots (1 + R'_n)(1 - \text{liquidity tax} / {\text{final value}})} - 1

After-Tax Excess Return

x=RBx' = R' - B'

where:

  • RR' = after-tax return of the portfolio
  • BB' = after-tax return of the benchmark

αtax=xx\alpha_{\text{tax}} = x' - x

where:

  • xx' = after-tax excess return
  • xx = pretax excess return

Tax-Efficiency Ratio (TER)

TER=RRTER = \frac{R'}{R}

where:

  • RR' = after-tax return
  • RR = pretax return

note:

  • for positive returns in taxable accounts, the higher the TER, the better.

Chapter: Trading, Performance Evaluation

Implementation Shortfall

IS (i.e. implementation shortfall) measures the total cost of trading, expressed as basis points of the total cost of the paper portfolio. e.g. buy order

IS=paper returnactual return\text{IS} = \text{paper return} - \text{actual return}

where:

  • paper return=total order size×(close pricedecision price)\text{paper return} = \text{total order size} \times (\text{close price} - \text{decision price})
  • actual return=filled order size×(close priceexecution price)commission\text{actual return} = \text{filled order size} \times (\text{close price} - \text{execution price}) - \text{commission}

Decomposition of IS. e.g. buy order

  • Execution cost = filled order size×(execution pricedecision price)\text{filled order size} \times (\text{execution price} - \text{decision price})
    • Delay cost = filled order size×(arrival pricedecision price)\text{filled order size} \times (\text{arrival price} - \text{decision price}):
    • trading cost (i.e. market impact cost) = filled order size×(execution pricearrival price)\text{filled order size} \times (\text{execution price} - \text{arrival price})
  • Opportunity cost = unfilled order size×(close pricedecision price)\text{unfilled order size} \times (\text{close price} - \text{decision price})
  • Fixed cost = filled order size×commission\text{filled order size} \times \text{commission}

note:

  • for sell order, flip the sign
  • execution risk: adverse price movement occurs over the trading horizon (in volatile market)
  • market impact cost: trade at more adverse prices to execute a larger transaction (in illiquid market)

Market-adjusted cost

market-adjusted cost removes the impact of market movements on trade cost.

market-adjusted cost(bps)=arrival cost(bps)β×index cost(bps)\text{market-adjusted cost} \text{(bps)} = \text{arrival cost} \text{(bps)} - \beta \times\text{index cost} \text{(bps)}

where:

  • arrival cost = side×(execution price/arrival price1)\text{side} \times (\text{execution price} / \text{arrival price} - 1)
  • index cost = side×(index average price/index arrival price1)\text{side} \times (\text{index average price} / \text{index arrival price} - 1)

Return Attribution

RA=i=1NΔWiRiR_A = \sum\limits_{i=1}^N{\Delta W_i}{R_i}

where:

  • RiR_i: return from security i
  • ΔWi\Delta W_i: active weight of security i, i.e. diff between portfolio and benchmark weight

BHB Method, i.e. Brinson-Hood-Beebower

Portfolio Return,R=i=1nwiRiBenchmark Return,B=i=1nWiBi\begin{aligned} \text{Portfolio Return}, R = \sum\limits_{i=1}^n{w_i}{R_i} \\ \text{Benchmark Return}, B = \sum\limits_{i=1}^n{W_i}{B_i} \end{aligned}

where:

  • wiw_i: portfolio weight of the ith segment (e.g. style, sector, geography, etc.)
  • RiR_i: portfolio return in the ith segment
  • WiW_i: benchmark weight of the ith segment
  • BiB_i: benchmark return in the ith segment
  • n: number of segments

RA=RB=i=1nwiRii=1nWiBi=i=1n(wiRiWiBi)=i=1n[(wiWi)BiAllocation Effect+Wi(RiBi)Selection Effect+(wiWi)(RiBi)Interaction Effect ]\begin{aligned} R_A & = R - B \\ & = \sum\limits_{i=1}^n{w_i}{R_i} - \sum\limits_{i=1}^n{W_i}{B_i} \\ & = \sum\limits_{i=1}^n(w_i R_i - W_i B_i) \\ & = \sum\limits_{i=1}^n[\underbrace{(w_i - W_i)\cdot B_i}_{\text{Allocation Effect}} + \underbrace{Wi \cdot (R_i - B_i)}_{\text{Selection Effect}} + \underbrace{(w_i - W_i) \cdot (R_i - B_i)}_{\text{Interaction Effect }}] \end{aligned}

BF Method, i.e. Brinson-Fachler

to solve BHB Method’s issue of misleading allocation effect, due to the sign of the resulting allocation effect does not automatically indicate whether the decision to overweight/underweight a particular segment of the portfolio was correct.

Allocation Effect=(wiWi)(BiB)\text{Allocation Effect} = (w_i - W_i)\cdot (B_i - \bm{B})

where:

  • B: overall benchmark return

note:

  • the additional term: i=1n(wiWi)(B)=(i=1nwii=1nWi)(B)=(11)(B)=0\sum\limits_{i=1}^n(w_i - W_i) \cdot (-B) = (\sum\limits_{i=1}^n{w_i} - \sum\limits_{i=1}^n{W_i}) \cdot (-B) = (1 - 1) \cdot (-B) = 0

Arithmetic vs Geometric Attribution

Period 1 Period 2
Portfolio Return ® 5% 5%
Benchmark Return (B) 3% 3%
  • arithmetic attribution: RA=RB=5%3%=2%R_A = R - B = 5\% - 3\% = 2\%
  • geometric attribution: RA=1+R1+B1=1.051.031=1.94%R_A = \frac{1 + R}{1 + B} - 1 = \frac{1.05}{1.03} - 1 = 1.94\%

geometric attribution does compound across periods: (1+1.94%)21=(1.05/1.03)21=3.92%(1 + 1.94\%)^2 - 1 = (1.05/1.03)^2 - 1 = 3.92\%

Performance Metrics

Sharpe Ratio

calculated as excess return over the risk-free rate (numerator) divided by standard deviation (denominator)

SA=RˉArˉfσ^AS_A = \frac{\bar{R}_A - \bar{r}_f}{\hat{\sigma}_A}

note:

  • denominator does not differentiate between volatility that is upside versus downside. Therefore, there is a penalty for all volatility, even if it is “good” volatility (addressed in Sortino Ratio below).

Treynor Ratio

similar to the Sharpe ratio, but the denominator is measured by beta, so only considering systematic risk rather than total risk

TA=RˉArˉfβ^AT_A = \frac{\bar{R}_A - \bar{r}_f}{\hat{\beta}_A}

note:

  • assume efficient markets
  • only useful in evaluating portfolios that have systematic risk and do not have unsystematic risk (i.e. well diversified)

Information Ratio

measures a portfolio’s performance against the benchmark but accounts for differences in risk

IR=E(rp)E(rB)σ(rprB)IR = \frac{E(r_p) - E(r_B)}{\sigma(r_p - r_B)}

note:

  • the denominator is known as the tracking risk, or the variability in the portfolio performance with that of its benchmark

Appraisal Ratio

measures the ratio of active return, α, to the volatility of the residual term, σεσ_ε, both derived from a factor-based regression

AR=ασεAR = \frac{α}{σ_ε}

note:

  • higher AR indicates generating more active return per unit of active risk (as represented by residual risk σεσ_ε also referred to as the “standard error of the regression”) and is therefore a superior active manager according to this measure.
  • σεσ_ε, i.e. the standard error of the regression, is the volatility of the error term in a factor-based regression, representing the component of the volatility of returns not explained by the regression model factors. This component of volatility is generated by the manager taking active bets relative to the factors in the model, hence σεσ_ε here is playing the role of active risk.
  • The AR is analogous to the IR. It looks at active return per unit of active risk. The only difference is that the AR uses a factor-based regression to estimate active return and active risk.

Sortino Ratio

only considers the standard deviation of the downside risk, in contrast to the Sharpe ratio, which considers all risk (e.g., both upside and downside). Positive volatility associated with the upside can be considered “good” volatility. As a result, the Sortino ratio can provide a more meaningful view of a portfolio’s risk-adjusted performance than the Sharpe ratio.

SRD=E(rp)rTσDSR_D = \frac{E(r_p) - r_T}{\sigma_D}

where:

  • rTr_T: target rate of return, or minimum acceptable return (MAR)
  • σD\sigma_D: target semideviation, measuring the standard deviation of returns below the target return

note:

  • determination of MAR is subjective and specific to each investor, causing comparability problems

Capture Ratio

Capture Ratio (CR) = UC ratio / DC ratio, a measure of return asymmetry, > 1 = positive asymmetry (convex shape), < 1 = negative asymmetry (concave shape)

Capture Ratio=Upside Capture RatioDownside Capture Ratio\text{Capture Ratio} = \frac{\text{Upside Capture Ratio}}{\text{Downside Capture Ratio}}

e.g.

  • if portfolio return is 5%, and benchmark return is 4%, then upside capture ratio 5% / 4% = 125%, > 100% indicates outperformed
  • if portfolio return is -3%, and benchmark return is -4%, then downside capture ratio -3% / -4% = 75% < 100% indicates outperformed
  • then the capture ratio is 125% / 75% = 1.67, indicates a positively asymmetrical (convex) return profile

note:

  • ideally, the manager would capture as much of the upside as possible and capture as little of the downside as possible to maximize the capture ratio
  • when betas are increasing (decreasing), momentum-driven strategies should have higher (lower) UC than value-driven strategies. A low-beta (high-beta) strategy will have lower (higher) UC and DC. Therefore, CRs can be used to confirm the investment strategy.

Drawdown

Month Monthly Return Drawdown Cumulative Drawdown Comments
01/2018 3.14% 0.00%
02/2018 –2.55% –2.55% –2.55% Drawdown phase begins
03/2018 –2.71% –2.71% –5.26%
04/2018 –4.66% –4.66% –9.92%
05/2018 –4.91% –4.91% –14.83%
06/2018 –0.73% –0.73% –15.56% Maximum drawdown
07/2018 2.18% –13.38% Recovery phase begins
08/2018 3.11% –10.27%
09/2018 2.45% –7.82%
10/2018 3.65% –4.17%
11/2018 4.03% –0.14%
12/2018 4.14% 0.00% Drawdown recovered

Time-to-Cash Table

liquidity classification schedule (time-to-cash table) is used to manage liquidity risk, including:

    1. amount of time needed to convert assets to cash
    1. liquidity classification level
    1. liquidity budget
Time to Cash Liquidity Classification Liquidity Budget (% of portfolio)
< 1 week Highly liquid At least 5%
< 1 quarter Liquid At least 25%
< 1 year Semi-liquid At least 40%
> 1 year Illiquid Up to 40%

Ethical and Professional Standards

GIPS return calculation

Rt to be evaluated each time an interim ECF occurs. Then sub-periods returns are geometrically linked to create a TWRR (i.e. time-weighted rate of return).

Rt=EVBVBVR_t = \frac{EV - BV}{BV}

RTWRR=(1+R1)×(1+R2)××(1+Rt)R_{TWRR} = (1+R_1) \times (1+R_2) \times \cdots \times (1+R_t)

Approximations when portfolios are not valued daily, and ECFs (i.e. external cash flows) are relatively small. The larger the ECFs and the more volatile the market, the greater the discrepancy will be between the true TWRR and the approximations.

Modified Dietz Return

RMD=EVBVECFBV+adjusted ECFR_{MD} = \frac{EV - BV - ECF}{BV + \text{adjusted ECF}}

adjusted ECF=ECF×weight\text{adjusted ECF} = \sum{\text{ECF} \times \text{weight}}

where:

  • weight is based on proportion of days in use, e.g. if the 1st ECF was received on Day 7, meaning that the manager had use of it for 23 days. Then the weight is 23/30.

note:

  • ECFs will be positive if cash inflow, and negative if cash outflow

MIRR

MIRR (i.e. modified internal rate of return)

EV=BV(1+r)+ECF(1+r)weightEV = BV(1+r) + \sum{ECF(1 + r)^{\text{weight}}}

solve for r

where:

  • weight is the same defined in the modified Dietz Return

Hexo, documentation, troubleshooting, GitHub Issues.

Hexo

Install & Setup

1
$ npm install -g hexo-cli

Initialize

1
2
3
$ hexo init <folder>
$ cd <folder>
$ npm install

File Structure:

1
2
3
4
5
6
7
8
9
10
|-- _config.yml # configuration file for hexo
|-- source
|-- _posts
|-- one_blog.md # blog file
|-- _drafts
|-- one_draft.md # draft file
|-- themes
|-- landscape # default theme
|-- ocean
|-- _config.yml # configuration file for ocean

Create a new post

1
$ hexo new "My New Post" / $ hexo n "My New Post"

More info: Writing

Run server

1
$ hexo server / $ hexo s

More info: Server

Generate static files

1
$ hexo generate / $ hexo g

More info: Generating

Deploy to remote sites

1
$ hexo deploy / $ hexo d

More info: Deployment

Deploy to github

1
$ yarn add hexo-deployer-git

Then, go to hexo’s _config.yml, add this to deploy session:

1
2
3
4
deploy:
type: git
repo: https://github.com/xyshell/xyshell.github.io.git
branch: master

Theme

Install & Setup & Update

1
$ git clone https://github.com/zhwangart/hexo-theme-ocean.git themes/ocean

Modify theme setting in hexo’s _config.yml to ocean:

1
theme: ocean

Further update:

1
2
cd themes/ocean
git pull

Theme Reference

Ocean

https://github.com/zhwangart/hexo-theme-ocean

Minos

https://github.com/ppoffice/hexo-theme-minos

Next

https://github.com/theme-next/hexo-theme-next

Installation and Configuration

Intall kubectl

  • configuration file location: $HOME/.kube/config.

  • handy when going from a local environment to a cluster in the cloud, or from one cluster to another, such as from development to production.

1
$ kubectl config use-context foobar

GKE Quickstart

  1. GKE Install

  2. GKE Quickstart

  3. quick start command

1
2
3
4
5
6
7
$ gcloud container clusters create linuxfoundation

$ gcloud container clusters list

$ kubectl get nodes # kubectl comes free from gcloud

$ gcloud container clusters delete linuxfoundation

Minikube

  1. github repo

  2. install

1
2
3
4
5
$ curl -Lo minikube https://storage.googleapis.com/minikube/releases/latest/minikube-darwin-amd64

$ chmod +x minikube

$ sudo mv minikube /usr/local/bin
  1. quick start command
1
2
3
$ minikube start

$ kubectl get nodes

Main Deployment configuration

  1. Single-node

With a single-node deployment, all the components run on the same server. This is great for testing, learning, and developing around Kubernetes.

  1. Single head node, multiple workers

Adding more workers, a single head node and multiple workers typically will consist of a single node etcd instance running on the head node with the API, the scheduler, and the controller-manager.

  1. Multiple head nodes with HA, multiple workers

Multiple head nodes in an HA configuration and multiple workers add more durability to the cluster. The API server will be fronted by a load balancer, the scheduler and the controller-manager will elect a leader (which is configured via flags). The etcd setup can still be single node.

  1. HA etcd, HA head nodes, multiple workers

The most advanced and resilient setup would be an HA etcd cluster, with HA head nodes and multiple workers. Also, etcd would run as a true cluster, which would provide HA and would run on nodes separate from the Kubernetes head nodes.

CSS selectors

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
h1,
h2 {
...;
} /* html tag selector, h1-h6 */

#id {
...;
} /* id selector */

.class {
...;
} /* class selector */

.class img {
...;
} /* descendant combinator */

.class:hover {
...;
} /* user action pseudo class */

.class:nth-of-type(odd) {
...;
} /* location pseudo class
nth-of-type: number or odd/even
first-of-type/last-of-type
nth-child
first-child/last-child
*/

#leftmenu > ul > li > a {
...;
} /* child combinator */

.sidebar + h2 {
...;
} /* adjacent sibling combinator */

.sidebar ~ h2 {
...;
} /* general sibling combinator */

CSS property reference

text style

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
color: white; /* default black */
font-size: 18px; /* default 16px */
font-weight: bold; /* default normal */
font-style: italic; /* default normal */

/* whitespace between lines of test */
line-height: 1.5; /* default 1, relative to font-size */
/* space between characters */
letter-spacing: 2px; /* default 0 */

/* visit cssfontstack.com to find more web-safe fonts. */
font-family: Verdana, Arial, sans-serif, cursive, fantasy; /* web-safe fonts, default Arial */
/* serif: Georgia, Times New Roman
sans-serif: Arial, Verdana, Comic Sans, Trebuchet
monospace: Courier New */

text-align: center; /* justify, default left */
text-transform: capitalize; /* uppercase, default none */

container style

The Box Model: element < padding < border < margin

Use chrome dev tools to inspect the box model.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
width: 100%;
/* relative to parent width, default auto
width: 1em; relative to parent font-size
width: 1rem; relative to html font-size */
backgroud-image: url(../images/image.png);
background-repeat: repeat; /* repeat, repeat-x, repeat-y, no-repeat */
background-color: #23cea6; /* RGB, default transparent */
border: 10px solid #a693c2;
border-color: #a693c2;
/* border inside, width, style, color(default text color)
style: solid, dashed, dotted, ridge, double, groove, inset, outset
border-left: xxx; left only */
outline: 10px solid red; /* border outside */

padding: 10px 20px 30px 40px;
/* space between container and content
top right bottom left
top/bottom left/right
top/right/bottom/left */
padding-left: 10px;

margin: 10px 20px 30px 40px;
/* space between container and next element
margin collapse, max(margin-bottom, margin-top) applied */
margin-bottom: 10px;

CSS templates

normalize.css

normalize.css v8.0.1 is a collection of CSS resets and normalizations.

Mobile first design

small.css

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
html {
-webkit-text-size-adjust: 100%;
-ms-text-size-adjust: 100%;
}

* {
-moz-box-sizing: border-box;
-webkit-box-sizing: border-box;
box-sizing: border-box;
}

body {
font-family: "Lato", sans-serif;
font-size: 16px;
}

medium.css

1
2
3
@media only screen and (min-width: 768px) {
...;
}

large.css

1
2
3
@media only screen and (min-width: 1024px) {
...;
}

IDE features

Multi-cursor editing

Using multiple cursors allows you to edit multiple parts of the document at once, greatly improving your productivity.

Vscode

  • Box Selection - ⇧↓, ⇧→, ⇧↑, ⇧← to select a block of text. ⇧⌥ while selecting text with the mouse or drag-select using the middle mouse button.
  • Add a cursor - ⌥⌘↑ to add a new cursor above, or ⌥⌘↓ to add a new cursor below. You can also use your mouse with ⌥+Click to add a cursor anywhere.
  • Create cursors on all occurrences of a string - select one instance of a string e.g. background-color and press ⇧⌘L. Now you can replace all instances by simply typing.
1
2
3
4
5
6
7
8
9
#p1 {
background-color: #ff0000;
} /* red in HEX format */
#p2 {
background-color: hsl(120, 100%, 50%);
} /* green in HSL format */
#p3 {
background-color: rgba(0, 4, 255, 0.733);
} /* blue with alpha channel in RGBA format */

IntelliSense

Vscode

  • invoke IntelliSense: ⌃Space or ⌥Esc
1
2
3
4
5
const canvas = document.querySelector('canvas');
const context = canvas.getContext('2d');

context.strokeStyle = 'blue';
context.

Line Actions

Vscode

  • Copy a line and insert it above or below: ⇧⌥↓ or ⇧⌥↑
  • Move an entire line or selection of lines up or down: ⌥↑ and ⌥↓
  • Delete the entire line: ⇧⌘K.
  • Comment out a line: ⌘/
1
2
3
4
5
{
"name": "John",
"age": 31,
"city": "New York"
}

Refactoring

Vscode

  • Rename a function/variable: F2 or right-click on the function/variable name.

Formatting

Vscode

  • Format a document: ⇧⌘F
  • Format a selection: ⌘K ⌘F

ps: enable editor.formatOnSave

Code Folding

Vscode

  • fold and unfold a block of code: ⌥⌘[ and ⌥⌘]
  • fold and unfold all blocks: ⌘K ⌘0 and ⌘K ⌘J
  • fold and unfold a number of levels: ⌘K ⌘1 to ⌘K ⌘5

Errors and Warnings

Vscode

  • navigate to the next error: F8

Code Snipets

Vscode

  • create a code snippet: Code > Preferences > User Snippets

CPA – C++ Certified Associate Programmer

Readings

More books: https://stackoverflow.com/questions/388242/the-definitive-c-book-guide-and-list

Reference

cout manipulator

1
2
3
4
int byte = 255;
cout << "Byte in hex: " << hex << byte;
cout << "Byte in decimal: " << dec << byte;
cout << "Byte in octal: " << oct << byte;
1
2
3
4
5
6
7
8
9
10
#include <iostream>
#include <iomanip>

using namespace std;
int main(void)
{
int byte = 255;
cout << setbase(16) << byte;
return 0;
}
1
2
3
float x = 2.5, y = 0.0000000025;
cout << fixed << x << " " << y << endl;
cout << scientific << x << " " << y << endl;

String methods

compare

str1.compare(str2)

  • str1.compare(str2) == 0 when str1 == str2

  • str1.compare(str2) > 0 when str1 > str2

  • str1.compare(str2) < 0 when str1 < str2

S.compare(substr_start, substr_length, other_string)

S.compare(substr_start, substr_length, other_string, other_substr_start, other_substr_length)

substr

1
string newstr = oldstr.substr(substring_start_position, length_of_substring)

length, size, capacity, max_size

1
2
3
4
5
6
7
int string_size = S.size();

int string_length = S.length();

int string_capacity = s.capacity();

int string_max_size = s.max_size();
1
TheString.reserve(100);

reserve, resize, clear, empty

1
2
3
4
5
6
7
bool is_empty = TheString.empty();

TheString.resize(50,'?');

TheString.resize(4);

TheString.clear();

find

1
2
3
int where_it_begins = S.find(another_string, start_here);

int where_it_is = S.find(any_character, start_here);
1
2
3
4
int comma = greeting.find(',');
if(comma != string::npos){
//found
};

append, push_back, insert

1
2
3
NewString.append(TheString);
NewString.append(TheString,0,3);
NewString.append(2,'!');
1
TheString.push_back(car);
1
2
string quote = "Whyserious?", anyword = "monsoon";
quote.insert(3,2,' ').insert(4,anyword,3,2); // Why so serious?

assign

1
2
string sky; 
sky.assign(80,'*');

replace

1
2
3
4
string ToDo = "I'll think about that in one hour"; 
string Schedule = "today yesterday tomorrow";

ToDo.replace(22, 12, Schedule, 16, 8); // I'll think about that tomorrow

erase

1
2
3
4
string WhereAreWe = "I've got a feeling we're not in Kansas anymore"; 

WhereAreWe.erase(38, 8).erase(25, 4); // I've got a feeling we're in Kansas
TheString.erase();

swap

1
2
3
4
string Drink = "A martini";
string Needs = "Shaken, not stirred";

Drink.swap(Needs);

This note follows Essential SQLAlchemy 2nd Edition

Github repo: https://github.com/oreillymedia/essential-sqlalchemy-2e/

SQLAlchemy Core

Create Schema

Relationship-visualization

Full in-memory SQLite code sample

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
from datetime import datetime
from sqlalchemy import (MetaData, Table, Column, Integer, Numeric, String, DateTime, ForeignKey, create_engine)

metadata = MetaData()
cookies = Table('cookies', metadata,
Column('cookie_id', Integer(), primary_key=True),
Column('cookie_name', String(50), index=True),
Column('cookie_recipe_url', String(255)),
Column('cookie_sku', String(55)),
Column('quantity', Integer()),
Column('unit_cost', Numeric(12, 2))
)

users = Table('users', metadata,
Column('user_id', Integer(), primary_key=True),
Column('customer_number', Integer(), autoincrement=True),
Column('username', String(15), nullable=False, unique=True),
Column('email_address', String(255), nullable=False),
Column('phone', String(20), nullable=False),
Column('password', String(25), nullable=False),
Column('created_on', DateTime(), default=datetime.now),
Column('updated_on', DateTime(), default=datetime.now, onupdate=datetime.now)
)

orders = Table('orders', metadata,
Column('order_id', Integer(), primary_key=True),
Column('user_id', ForeignKey('users.user_id'))
)

line_items = Table('line_items', metadata,
Column('line_items_id', Integer(), primary_key=True),
Column('order_id', ForeignKey('orders.order_id')),
Column('cookie_id', ForeignKey('cookies.cookie_id')),
Column('quantity', Integer()),
Column('extended_cost', Numeric(12, 2))
)

engine = create_engine('sqlite:///:memory:')
metadata.create_all(engine)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
from datetime import datetime

from sqlalchemy import (MetaData, Table, Column, Integer, Numeric, String, DateTime, ForeignKey, Boolean, create_engine, CheckConstraint)
metadata = MetaData()

cookies = Table('cookies', metadata,
Column('cookie_id', Integer(), primary_key=True),
Column('cookie_name', String(50), index=True),
Column('cookie_recipe_url', String(255)),
Column('cookie_sku', String(55)),
Column('quantity', Integer()),
Column('unit_cost', Numeric(12, 2)),
CheckConstraint('quantity >= 0', name='quantity_positive')
)

users = Table('users', metadata,
Column('user_id', Integer(), primary_key=True),
Column('username', String(15), nullable=False, unique=True),
Column('email_address', String(255), nullable=False),
Column('phone', String(20), nullable=False),
Column('password', String(25), nullable=False),
Column('created_on', DateTime(), default=datetime.now),
Column('updated_on', DateTime(), default=datetime.now, onupdate=datetime.now)
)

orders = Table('orders', metadata,
Column('order_id', Integer()),
Column('user_id', ForeignKey('users.user_id')),
Column('shipped', Boolean(), default=False)
)

line_items = Table('line_items', metadata,
Column('line_items_id', Integer(), primary_key=True),
Column('order_id', ForeignKey('orders.order_id')),
Column('cookie_id', ForeignKey('cookies.cookie_id')),
Column('quantity', Integer()),
Column('extended_cost', Numeric(12, 2))
)

engine = create_engine('sqlite:///:memory:')
metadata.create_all(engine)
connection = engine.connect()

Data Operation

Insert

Way1:

1
2
3
4
5
6
7
8
9
10
ins = cookies.insert().values(
cookie_name="chocolate chip",
cookie_recipe_url="http://some.aweso.me/cookie/recipe.html",
cookie_sku="CC01",
quantity="12",
unit_cost="0.50"
)
# str(ins)
result = connection.execute(ins)
# result.inserted_primary_key

Way2:

1
2
3
4
5
6
7
8
from sqlalchemy import insert
ins = insert(cookies).values(
cookie_name="chocolate chip",
cookie_recipe_url="http://some.aweso.me/cookie/recipe.html",
cookie_sku="CC01",
quantity="12",
unit_cost="0.50"
)

Way3:

1
2
3
4
5
6
7
8
ins = cookies.insert()
result = connection.execute(
ins,
cookie_name='dark chocolate chip',
cookie_recipe_url='http://some.aweso.me/cookie/recipe_dark.html',
cookie_sku='CC02',
quantity='1',
unit_cost='0.75')

Multiple:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
inventory_list = [
{
'cookie_name': 'peanut butter',
'cookie_recipe_url': 'http://some.aweso.me/cookie/peanut.html',
'cookie_sku': 'PB01',
'quantity': '24',
'unit_cost': '0.25'
},
{
'cookie_name': 'oatmeal raisin',
'cookie_recipe_url': 'http://some.okay.me/cookie/raisin.html',
'cookie_sku': 'EWW01',
'quantity': '100',
'unit_cost': '1.00'
}
]
result = connection.execute(ins, inventory_list)

Select

Way1:

1
2
3
4
5
from sqlalchemy.sql import select
s = select([cookies])
# str(s)
rp = connection.execute(s)
results = rp.fetchall()

Way2:

1
2
3
4
s = cookies.select()
rp = connection.execute(s)
for record in rp:
print(record.cookie_name)

Sepcify columns, order by, limit, cast:

1
2
3
4
5
6
7
8
9
10
11
from sqlalchemy import desc
from sqlalchemy import cast

s = select([cookies.c.cookie_name, cookies.c.quantity, 'SKU-' + cookies.c.cookie_sku,
cast((cookies.c.quantity * cookies.c.unit_cost), Numeric(12,2)).label('inv_cost')]])
s = s.order_by(desc(cookies.c.quantity), cookies.c.cookie_name)
s = s.limit(2) # Also by, s = cookies.select(limit=1)
rp = connection.execute(s)
# rp.keys()
for cookie in rp:
print('{} - {}'.format(cookie.quantity, cookie.cookie_name))

Aggregate:

1
2
3
4
5
6
7
from sqlalchemy.sql import func

s = select([func.count(cookies.c.cookie_name).label('inventory_count')])
rp = connection.execute(s)
record = rp.first()
# record.keys()
# record.inventory_count

Filter(Where):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
s = select([cookies]).where(cookies.c.cookie_name == 'chocolate chip')
rp = connection.execute(s)
record = rp.first()
# record.items()

s = select([cookies]).where(cookies.c.cookie_name.like('%chocolate%')).where(cookies.c.quantity == 12)
# str(s)
rp = connection.execute(s)
for record in rp.fetchall():
print(record.cookie_name)

from sqlalchemy import and_, or_, not_
s = select([cookies]).where(or_(
cookies.c.quantity.between(10, 50),
cookies.c.cookie_name.contains('chip')
))
for row in connection.execute(s):
print(row.cookie_name)

Update

1
2
3
4
5
from sqlalchemy import update
u = update(cookies).where(cookies.c.cookie_name == "chocolate chip")
u = u.values(quantity=(cookies.c.quantity + 120))
result = connection.execute(u)
print(result.rowcount)

Delete

1
2
3
4
from sqlalchemy import delete
u = delete(cookies).where(cookies.c.cookie_name == "dark chocolate chip")
result = connection.execute(u)
print(result.rowcount)

Join

1
2
3
4
5
6
columns = [orders.c.order_id, users.c.username, users.c.phone, cookies.c.cookie_name, line_items.c.quantity, line_items.c.extended_cost]
cookiemon_orders = select(columns)
cookiemon_orders = cookiemon_orders.select_from(users.join(orders).join(line_items).join(cookies)).where(users.c.username == 'cookiemon')
result = connection.execute(cookiemon_orders).fetchall()
for row in result:
print(row)

outerjoin:

1
2
3
4
5
6
columns = [users.c.username, orders.c.order_id]
all_orders = select(columns)
all_orders = all_orders.select_from(users.outerjoin(orders))
result = connection.execute(all_orders).fetchall()
for row in result:
print(row)

Alias

1
2
3
manager = employee_table.alias()
stmt = select([employee_table.c.name], and_(employee_table.c.manager_id==manager.c.id, manager.c.name=='Fred'))
print(stmt)

Groupby

1
2
3
4
5
6
7
columns = [users.c.username, func.count(orders.c.order_id)]
all_orders = select(columns)
all_orders = all_orders.select_from(users.outerjoin(orders)).group_by(users.c.username)
print(str(all_orders))
result = connection.execute(all_orders).fetchall()
for row in result:
print(row)

Raw Queries

1
result = connection.execute("select * from orders").fetchall()
1
2
3
from sqlalchemy import text
stmt = select([users]).where(text('username="cookiemon"'))
print(connection.execute(stmt).fetchall())

Exception and Transaction

Transaction Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
# Use second schema

from sqlalchemy import select, insert, update
ins = insert(users).values(
username="cookiemon",
email_address="mon@cookie.com",
phone="111-111-1111",
password="password"
)
result = connection.execute(ins)

ins = cookies.insert()
inventory_list = [
{
'cookie_name': 'chocolate chip',
'cookie_recipe_url': 'http://some.aweso.me/cookie/recipe.html',
'cookie_sku': 'CC01',
'quantity': '12',
'unit_cost': '0.50'
},
{
'cookie_name': 'dark chocolate chip',
'cookie_recipe_url': 'http://some.aweso.me/cookie/recipe_dark.html',
'cookie_sku': 'CC02',
'quantity': '1',
'unit_cost': '0.75'
}
]
result = connection.execute(ins, inventory_list)

ins = insert(orders).values(user_id=1, order_id=1)
result = connection.execute(ins)
ins = insert(line_items)
order_items = [
{
'order_id': 1,
'cookie_id': 1,
'quantity': 9,
'extended_cost': 4.50
}
]
result = connection.execute(ins, order_items)


ins = insert(orders).values(user_id=1, order_id=2)
result = connection.execute(ins)
ins = insert(line_items)
order_items = [
{
'order_id': 2,
'cookie_id': 2,
'quantity': 1,
'extended_cost': 1.50
},
{
'order_id': 2,
'cookie_id': 1,
'quantity': 4,
'extended_cost': 4.50
}
]
result = connection.execute(ins, order_items)

def ship_it(order_id):

s = select([line_items.c.cookie_id, line_items.c.quantity])
s = s.where(line_items.c.order_id == order_id)
cookies_to_ship = connection.execute(s)
for cookie in cookies_to_ship:
u = update(cookies).where(cookies.c.cookie_id == cookie.cookie_id)
u = u.values(quantity = cookies.c.quantity - cookie.quantity)
result = connection.execute(u)
u = update(orders).where(orders.c.order_id == order_id)
u = u.values(shipped=True)
result = connection.execute(u)
print("Shipped order ID: {}".format(order_id))

ship_it(1)

s = select([cookies.c.cookie_name, cookies.c.quantity])
connection.execute(s).fetchall()

ship_it(2)

u = update(cookies).where(cookies.c.cookie_name == "dark chocolate chip")
u = u.values(quantity = 1)
result = connection.execute(u)

from sqlalchemy.exc import IntegrityError
def ship_it(order_id):
s = select([line_items.c.cookie_id, line_items.c.quantity])
s = s.where(line_items.c.order_id == order_id)
transaction = connection.begin()
cookies_to_ship = connection.execute(s).fetchall()
try:
for cookie in cookies_to_ship:
u = update(cookies).where(cookies.c.cookie_id == cookie.cookie_id)
u = u.values(quantity = cookies.c.quantity-cookie.quantity)
result = connection.execute(u)
u = update(orders).where(orders.c.order_id == order_id)
u = u.values(shipped=True)
result = connection.execute(u)
print("Shipped order ID: {}".format(order_id))
transaction.commit()
except IntegrityError as error:
transaction.rollback()
print(error)
ship_it(2)

s = select([cookies.c.cookie_name, cookies.c.quantity])
connection.execute(s).fetchall()

Reflection

Reflecting Individual Tables

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
from sqlalchemy import MetaData, Table, create_engine
metadata = MetaData()
engine = create_engine('sqlite:///Chinook_Sqlite.sqlite')


artist = Table('artist', metadata, autoload=True, autoload_with=engine)
album = Table('album', metadata, autoload=True, autoload_with=engine)

artist.columns.keys()
from sqlalchemy import select
s = select([artist]).limit(10)
engine.execute(s).fetchall()

album.foreign_keys
from sqlalchemy import ForeignKeyConstraint
album.append_constraint(
ForeignKeyConstraint(['ArtistId'], ['artist.ArtistId'])
)
metadata.tables['album']
str(artist.join(album))

Reflecting a Whole Database

1
2
3
4
5
6
7
8
metadata.reflect(bind=engine)

metadata.tables.keys()

playlist = metadata.tables['Playlist']
from sqlalchemy import select
s = select([playlist]).limit(10)
engine.execute(s).fetchall()

SQLAlchemy ORM

Defining Tables via ORM Classes

sql_orm_setup

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
from datetime import datetime

from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy import Table, Column, Integer, Numeric, String, Boolean, DateTime, ForeignKey
from sqlalchemy import create_engine
from sqlalchemy.orm import relationship, backref

Base = declarative_base()


class Cookie(Base):
__tablename__ = 'cookies'

cookie_id = Column(Integer(), primary_key=True, )
cookie_name = Column(String(50), index=True)
cookie_recipe_url = Column(String(255))
cookie_sku = Column(String(55))
quantity = Column(Integer())
unit_cost = Column(Numeric(12, 2))

def __repr__(self):
return "Cookie(cookie_name='{self.cookie_name}', " \
"cookie_recipe_url='{self.cookie_recipe_url}', " \
"cookie_sku='{self.cookie_sku}', " \
"quantity={self.quantity}, " \
"unit_cost={self.unit_cost})".format(self=self)


class User(Base):
__tablename__ = 'users'

user_id = Column(Integer(), primary_key=True)
username = Column(String(15), nullable=False, unique=True)
email_address = Column(String(255), nullable=False)
phone = Column(String(20), nullable=False)
password = Column(String(25), nullable=False)
created_on = Column(DateTime(), default=datetime.now)
updated_on = Column(DateTime(), default=datetime.now,
onupdate=datetime.now)

def __repr__(self):
return "User(username='{self.username}', " \
"email_address='{self.email_address}', " \
"phone='{self.phone}', " \
"password='{self.password}')".format(self=self)


class Order(Base):
__tablename__ = 'orders'
order_id = Column(Integer(), primary_key=True)
user_id = Column(Integer(), ForeignKey('users.user_id'))
shipped = Column(Boolean(), default=False)
user = relationship("User", backref=backref(
'orders', order_by=order_id)) # one-to-many relationship

def __repr__(self):
return "Order(user_id={self.user_id}, " \
"shipped={self.shipped})".format(self=self)


class LineItem(Base):
__tablename__ = 'line_items'
line_items_id = Column(Integer(), primary_key=True)
order_id = Column(Integer(), ForeignKey('orders.order_id'))
cookie_id = Column(Integer(), ForeignKey('cookies.cookie_id'))
quantity = Column(Integer())
extended_cost = Column(Numeric(12, 2))
order = relationship("Order", backref=backref(
'line_items', order_by=line_items_id))
cookie = relationship("Cookie", uselist=False) # one-to-one relationship

def __repr__(self):
return "LineItems(order_id={self.order_id}, " \
"cookie_id={self.cookie_id}, " \
"quantity={self.quantity}, " \
"extended_cost={self.extended_cost})".format(
self=self)


engine = create_engine('sqlite:///sql_orm_setup.sqlite', echo=True)
Base.metadata.create_all(engine)

sql_orm_setup_cons

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
import os
from sqlalchemy.orm import relationship, backref
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy import Table, Column, Integer, Numeric, String, DateTime, ForeignKey, Boolean, CheckConstraint
from datetime import datetime
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker

dbname = os.path.basename(__file__).replace('.py', '.sqlite')
engine = create_engine('sqlite:///{}'.format(dbname), echo=True)

Session = sessionmaker(bind=engine)

session = Session()


Base = declarative_base()


class Cookie(Base):
__tablename__ = 'cookies'
__table_args__ = (CheckConstraint(
'quantity >= 0', name='quantity_positive'),)

cookie_id = Column(Integer, primary_key=True)
cookie_name = Column(String(50), index=True)
cookie_recipe_url = Column(String(255))
cookie_sku = Column(String(55))
quantity = Column(Integer())
unit_cost = Column(Numeric(12, 2))

def __init__(self, name, recipe_url=None, sku=None, quantity=0, unit_cost=0.00):
self.cookie_name = name
self.cookie_recipe_url = recipe_url
self.cookie_sku = sku
self.quantity = quantity
self.unit_cost = unit_cost

def __repr__(self):
return "Cookie(cookie_name='{self.cookie_name}', " \
"cookie_recipe_url='{self.cookie_recipe_url}', " \
"cookie_sku='{self.cookie_sku}', " \
"quantity={self.quantity}, " \
"unit_cost={self.unit_cost})".format(self=self)


class User(Base):
__tablename__ = 'users'

user_id = Column(Integer(), primary_key=True)
username = Column(String(15), nullable=False, unique=True)
email_address = Column(String(255), nullable=False)
phone = Column(String(20), nullable=False)
password = Column(String(25), nullable=False)
created_on = Column(DateTime(), default=datetime.now)
updated_on = Column(DateTime(), default=datetime.now,
onupdate=datetime.now)

def __init__(self, username, email_address, phone, password):
self.username = username
self.email_address = email_address
self.phone = phone
self.password = password

def __repr__(self):
return "User(username='{self.username}', " \
"email_address='{self.email_address}', " \
"phone='{self.phone}', " \
"password='{self.password}')".format(self=self)


class Order(Base):
__tablename__ = 'orders'
order_id = Column(Integer(), primary_key=True)
user_id = Column(Integer(), ForeignKey('users.user_id'))
shipped = Column(Boolean(), default=False)

user = relationship("User", backref=backref('orders', order_by=order_id))

def __repr__(self):
return "Order(user_id={self.user_id}, " \
"shipped={self.shipped})".format(self=self)


class LineItem(Base):
__tablename__ = 'line_items'
line_item_id = Column(Integer(), primary_key=True)
order_id = Column(Integer(), ForeignKey('orders.order_id'))
cookie_id = Column(Integer(), ForeignKey('cookies.cookie_id'))
quantity = Column(Integer())
extended_cost = Column(Numeric(12, 2))

order = relationship("Order", backref=backref(
'line_items', order_by=line_item_id))
cookie = relationship("Cookie", uselist=False)

def __repr__(self):
return "LineItems(order_id={self.order_id}, " \
"cookie_id={self.cookie_id}, " \
"quantity={self.quantity}, " \
"extended_cost={self.extended_cost})".format(
self=self)


Base.metadata.create_all(engine)


cookiemon = User('cookiemon', 'mon@cookie.com', '111-111-1111', 'password')
cc = Cookie('chocolate chip', 'http://some.aweso.me/cookie/recipe.html', 'CC01', 12, 0.50)
dcc = Cookie('dark chocolate chip',
'http://some.aweso.me/cookie/recipe_dark.html',
'CC02',
1,
0.75)
session.add(cookiemon)
session.add(cc)
session.add(dcc)


o1 = Order()
o1.user = cookiemon
session.add(o1)

line1 = LineItem(order=o1, cookie=cc, quantity=9, extended_cost=4.50)


session.add(line1)
session.commit()
o2 = Order()
o2.user = cookiemon
session.add(o2)

line1 = LineItem(order=o2, cookie=cc, quantity=2, extended_cost=1.50)
line2 = LineItem(order=o2, cookie=dcc, quantity=9, extended_cost=6.75)


session.add(line1)
session.add(line2)
session.commit()

sql_orm_setup_test

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
import os
from datetime import datetime

from sqlalchemy import (Column, Integer, Numeric, String, DateTime, ForeignKey,
Boolean, create_engine)
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import relationship, backref, sessionmaker


conn_string = os.path.basename(__file__).replace('.py', '.sqlite')
Base = declarative_base()


class Cookie(Base):
__tablename__ = 'cookies'

cookie_id = Column(Integer, primary_key=True)
cookie_name = Column(String(50), index=True)
cookie_recipe_url = Column(String(255))
cookie_sku = Column(String(55))
quantity = Column(Integer())
unit_cost = Column(Numeric(12, 2))

def __repr__(self):
return "Cookie(cookie_name='{self.cookie_name}', " \
"cookie_recipe_url='{self.cookie_recipe_url}', " \
"cookie_sku='{self.cookie_sku}', " \
"quantity={self.quantity}, " \
"unit_cost={self.unit_cost})".format(self=self)


class User(Base):
__tablename__ = 'users'

user_id = Column(Integer(), primary_key=True)
username = Column(String(15), nullable=False, unique=True)
email_address = Column(String(255), nullable=False)
phone = Column(String(20), nullable=False)
password = Column(String(25), nullable=False)
created_on = Column(DateTime(), default=datetime.now)
updated_on = Column(DateTime(), default=datetime.now, onupdate=datetime.now)

def __repr__(self):
return "User(username='{self.username}', " \
"email_address='{self.email_address}', " \
"phone='{self.phone}', " \
"password='{self.password}')".format(self=self)


class Order(Base):
__tablename__ = 'orders'
order_id = Column(Integer(), primary_key=True)
user_id = Column(Integer(), ForeignKey('users.user_id'))
shipped = Column(Boolean(), default=False)

user = relationship("User", backref=backref('orders', order_by=order_id))

def __repr__(self):
return "Order(user_id={self.user_id}, " \
"shipped={self.shipped})".format(self=self)


class LineItem(Base):
__tablename__ = 'line_items'
line_item_id = Column(Integer(), primary_key=True)
order_id = Column(Integer(), ForeignKey('orders.order_id'))
cookie_id = Column(Integer(), ForeignKey('cookies.cookie_id'))
quantity = Column(Integer())
extended_cost = Column(Numeric(12, 2))

order = relationship("Order", backref=backref('line_items',
order_by=line_item_id))
cookie = relationship("Cookie", uselist=False)

def __repr__(self):
return "LineItems(order_id={self.order_id}, " \
"cookie_id={self.cookie_id}, " \
"quantity={self.quantity}, " \
"extended_cost={self.extended_cost})".format(
self=self)


class DataAccessLayer:

def __init__(self):
self.engine = None
self.session = None
self.conn_string = conn_string

def connect(self):
self.engine = create_engine(self.conn_string)
Base.metadata.create_all(self.engine)
self.Session = sessionmaker(bind=self.engine)


dal = DataAccessLayer()


def prep_db(session):
c1 = Cookie(cookie_name='dark chocolate chip',
cookie_recipe_url='http://some.aweso.me/cookie/dark_cc.html',
cookie_sku='CC02',
quantity=1,
unit_cost=0.75)
c2 = Cookie(cookie_name='peanut butter',
cookie_recipe_url='http://some.aweso.me/cookie/peanut.html',
cookie_sku='PB01',
quantity=24,
unit_cost=0.25)
c3 = Cookie(cookie_name='oatmeal raisin',
cookie_recipe_url='http://some.okay.me/cookie/raisin.html',
cookie_sku='EWW01',
quantity=100,
unit_cost=1.00)
session.bulk_save_objects([c1, c2, c3])
session.commit()

cookiemon = User(username='cookiemon',
email_address='mon@cookie.com',
phone='111-111-1111',
password='password')
cakeeater = User(username='cakeeater',
email_address='cakeeater@cake.com',
phone='222-222-2222',
password='password')
pieperson = User(username='pieperson',
email_address='person@pie.com',
phone='333-333-3333',
password='password')
session.add(cookiemon)
session.add(cakeeater)
session.add(pieperson)
session.commit()

o1 = Order()
o1.user = cookiemon
session.add(o1)

line1 = LineItem(cookie=c1, quantity=2, extended_cost=1.00)

line2 = LineItem(cookie=c3, quantity=12, extended_cost=3.00)

o1.line_items.append(line1)
o1.line_items.append(line2)
session.commit()

o2 = Order()
o2.user = cakeeater

line1 = LineItem(cookie=c1, quantity=24, extended_cost=12.00)
line2 = LineItem(cookie=c3, quantity=6, extended_cost=6.00)

o2.line_items.append(line1)
o2.line_items.append(line2)

session.add(o2)
session.commit()

Data Operation

Session

1
2
3
4
5
from sqlalchemy.orm import sessionmaker

Session = sessionmaker(bind=engine)

session = Session()

Insert

Inserting a single object:

1
2
3
4
5
6
7
8
9
cc_cookie = Cookie(cookie_name='chocolate chip',
cookie_recipe_url='http://some.aweso.me/cookie/recipe.html',
cookie_sku='CC01',
quantity=12,
unit_cost=0.50)
session.add(cc_cookie)
session.commit()

cc_cookie.cookie_id

Multiple inserts:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
dcc = Cookie(cookie_name='dark chocolate chip',
cookie_recipe_url='http://some.aweso.me/cookie/recipe_dark.html',
cookie_sku='CC02',
quantity=1,
unit_cost=0.75)
mol = Cookie(cookie_name='molasses',
cookie_recipe_url='http://some.aweso.me/cookie/recipe_molasses.html',
cookie_sku='MOL01',
quantity=1,
unit_cost=0.80)
session.add(dcc)
session.add(mol)
session.flush()

print(dcc.cookie_id)
print(mol.cookie_id)

Bulk inserting multiple records:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
c1 = Cookie(cookie_name='peanut butter',
cookie_recipe_url='http://some.aweso.me/cookie/peanut.html',
cookie_sku='PB01',
quantity=24,
unit_cost=0.25)
c2 = Cookie(cookie_name='oatmeal raisin',
cookie_recipe_url='http://some.okay.me/cookie/raisin.html',
cookie_sku='EWW01',
quantity=100,
unit_cost=1.00)
session.bulk_save_objects([c1,c2])
session.commit()
# If you are inserting multiple records and don’t need access to relationships or the inserted primary key, use bulk_save_objects or its related methods.
c1.cookie_id

Query

1
2
3
4
5
cookies = session.query(Cookie).all()
print(cookies)

for cookie in session.query(Cookie):
print(cookie)

Controlling the Columns in the Query:

1
print(session.query(Cookie.cookie_name, Cookie.quantity).first())

Ordering:

1
2
3
4
5
6
7
for cookie in session.query(Cookie).order_by(Cookie.quantity):
print('{:3} - {}'.format(cookie.quantity, cookie.cookie_name))


from sqlalchemy import desc
for cookie in session.query(Cookie).order_by(desc(Cookie.quantity)):
print('{:3} - {}'.format(cookie.quantity, cookie.cookie_name))

Limit:

1
2
3
4
5
query = session.query(Cookie).order_by(Cookie.quantity)[:2] # inefficient
print([result.cookie_name for result in query])

query = session.query(Cookie).order_by(Cookie.quantity).limit(2)
print([result.cookie_name for result in query])

Built-In SQL Functions and Labels:

1
2
3
4
5
6
7
8
9
10
11
from sqlalchemy import func
inv_count = session.query(func.sum(Cookie.quantity)).scalar()
print(inv_count)

rec_count = session.query(func.count(Cookie.cookie_name)).first()
print(rec_count)

rec_count = session.query(func.count(Cookie.cookie_name) \
.label('inventory_count')).first()
print(rec_count.keys())
print(rec_count.inventory_count)

Filter:

1
2
3
4
5
6
7
8
9
record = session.query(Cookie).filter(Cookie.cookie_name == 'chocolate chip').first()
print(record)

record = session.query(Cookie).filter_by(cookie_name='chocolate chip').first()
print(record)

query = session.query(Cookie).filter(Cookie.cookie_name.like('%chocolate%'))
for record in query:
print(record.cookie_name)

Operators:

1
2
3
4
5
6
7
8
9
10
11
results = session.query(Cookie.cookie_name, 'SKU-' + Cookie.cookie_sku).all()
for row in results:
print(row)


from sqlalchemy import cast
query = session.query(Cookie.cookie_name,
cast((Cookie.quantity * Cookie.unit_cost),
Numeric(12,2)).label('inv_cost'))
for result in query:
print('{} - {}'.format(result.cookie_name, result.inv_cost))

Boolean Operators:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
from sqlalchemy import and_, or_, not_
query = session.query(Cookie).filter(
Cookie.quantity > 23,
Cookie.unit_cost < 0.40
)
for result in query:
print(result.cookie_name)


from sqlalchemy import and_, or_, not_
query = session.query(Cookie).filter(
or_(
Cookie.quantity.between(10, 50),
Cookie.cookie_name.contains('chip')
)
)
for result in query:
print(result.cookie_name)

Update

1
2
3
4
5
query = session.query(Cookie)
cc_cookie = query.filter(Cookie.cookie_name == "chocolate chip").first()
cc_cookie.quantity = cc_cookie.quantity + 120
session.commit()
print(cc_cookie.quantity)
1
2
3
4
5
6
query = session.query(Cookie)
query = query.filter(Cookie.cookie_name == "chocolate chip")
query.update({Cookie.quantity: Cookie.quantity - 20})

cc_cookie = query.first()
print(cc_cookie.quantity)

Delete

1
2
3
4
5
6
7
query = session.query(Cookie)
query = query.filter(Cookie.cookie_name == "dark chocolate chip")
dcc_cookie = query.one()
session.delete(dcc_cookie)
session.commit()
dcc_cookie = query.first()
print(dcc_cookie)
1
2
3
4
5
query = session.query(Cookie)
query = query.filter(Cookie.cookie_name == "molasses")
query.delete()
mol_cookie = query.first()
print(mol_cookie)

setup:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
cookiemon = User(username='cookiemon',
email_address='mon@cookie.com',
phone='111-111-1111',
password='password')
cakeeater = User(username='cakeeater',
email_address='cakeeater@cake.com',
phone='222-222-2222',
password='password')
pieperson = User(username='pieperson',
email_address='person@pie.com',
phone='333-333-3333',
password='password')
session.add(cookiemon)
session.add(cakeeater)
session.add(pieperson)
session.commit()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
o1 = Order()
o1.user = cookiemon
session.add(o1)

cc = session.query(Cookie).filter(Cookie.cookie_name =="chocolate chip").one()
line1 = LineItem(cookie=cc, quantity=2, extended_cost=1.00)

pb = session.query(Cookie).filter(Cookie.cookie_name =="peanut butter").one()
line2 = LineItem(quantity=12, extended_cost=3.00)
line2.cookie = pb
line2.order = o1

o1.line_items.append(line1)
o1.line_items.append(line2)
session.commit()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
o2 = Order()
o2.user = cakeeater

cc = session.query(Cookie).filter(Cookie.cookie_name == "chocolate chip").one()
line1 = LineItem(cookie=cc, quantity=24, extended_cost=12.00)

oat = session.query(Cookie).filter(Cookie.cookie_name == "oatmeal raisin").one()
line2 = LineItem(cookie=oat, quantity=6, extended_cost=6.00)

o2.line_items.append(line1)
o2.line_items.append(line2)

session.add(o2)
session.commit()

Joins

Using join to select from multiple tables:

1
2
3
4
query = session.query(Order.order_id, User.username, User.phone, Cookie.cookie_name, LineItem.quantity, LineItem.extended_cost)
query = query.join(User).join(LineItem).join(Cookie)
results = query.filter(User.username == 'cookiemon').all()
print(results)

Using outerjoin to select from multiple tables:

1
2
3
4
query = session.query(User.username, func.count(Order.order_id))
query = query.outerjoin(Order).group_by(User.username)
for row in query:
print(row)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
class Employee(Base):
__tablename__ = 'employees'

id = Column(Integer(), primary_key=True)
manager_id = Column(Integer(), ForeignKey('employees.id'))
name = Column(String(255), nullable=False)

manager = relationship("Employee", backref=backref('reports'), remote_side=[id])

Base.metadata.create_all(engine)

marsha = Employee(name='Marsha')
fred = Employee(name='Fred')
marsha.reports.append(fred)
session.add(marsha)
session.commit()

for report in marsha.reports:
print(report.name)

Group

1
2
3
4
query = session.query(User.username, func.count(Order.order_id))
query = query.outerjoin(Order).group_by(User.username)
for row in query:
print(row)

Chain

Chaining:

1
2
3
4
5
6
7
def get_orders_by_customer(cust_name):
query = session.query(Order.order_id, User.username, User.phone, Cookie.cookie_name, LineItem.quantity,LineItem.extended_cost)
query = query.join(User).join(LineItem).join(Cookie)
results = query.filter(User.username == cust_name).all()
return results

get_orders_by_customer('cakeeater')

Conditional chaining:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
def get_orders_by_customer(cust_name, shipped=None, details=False):
query = session.query(Order.order_id, User.username, User.phone)
query = query.join(User)
if details:
query = query.add_columns(Cookie.cookie_name, LineItem.quantity,
LineItem.extended_cost)
query = query.join(LineItem).join(Cookie)
if shipped is not None:
query = query.filter(Order.shipped == shipped)
results = query.filter(User.username == cust_name).all()
return results

print(get_orders_by_customer('cakeeater'))

print(get_orders_by_customer('cakeeater', details=True))

print(get_orders_by_customer('cakeeater', shipped=True))

print(get_orders_by_customer('cakeeater', shipped=False))

print(get_orders_by_customer('cakeeater', shipped=False, details=True))

Raw Queries

1
2
3
from sqlalchemy import text
query = session.query(User).filter(text("username='cookiemon'"))
print(query.all())

Session and Exceptions

from sql_orm_setup import *

Session States

1
2
3

cc_cookie = Cookie(
cookie_name='chocolate chip', cookie_recipe_url='http://some.aweso.me/cookie/recipe.html', cookie_sku='CC01', quantity=12, unit_cost=0.50)

transient:

1
2
3
4
from sqlalchemy import inspect
insp = inspect(cc_cookie)
for state in ['transient', 'pending', 'persistent', 'detached']:
print('{:>10}: {}'.format(state, getattr(insp, state)))

pending:

1
2
3
session.add(cc_cookie)
for state in ['transient','pending','persistent','detached']:
print('{:>10}: {}'.format(state, getattr(insp, state)))

persistent:

1
2
3
session.commit()
for state in ['transient','pending','persistent','detached']:
print('{:>10}: {}'.format(state, getattr(insp, state)))

detached:

1
2
3
session.expunge(cc_cookie)
for state in ['transient','pending','persistent','detached']:
print('{:>10}: {}'.format(state, getattr(insp, state)))
1
2
3
4
5
6
7
8
session.add(cc_cookie)
cc_cookie.cookie_name = 'Change chocolate chip'

insp.modified
for attr, attr_state in insp.attrs.items():
if attr_state.history.has_changes():
print('{}: {}'.format(attr, attr_state.value))
print('History: {}\n'.format(attr_state.history))

MultipleResultsFound Exception

1
2
3
4
5
6
7
8
9
10
11
12
13
dcc = Cookie('dark chocolate chip',
'http://some.aweso.me/cookie/recipe_dark.html',
'CC02', 1, 0.75)
session.add(dcc)
session.commit()

from sqlalchemy.orm.exc import MultipleResultsFound
try:
results = session.query(Cookie).one()
except MultipleResultsFound as exc:
print('We found too many cookies... is that even possible?')

session.query(Cookie).all()

DetachedInstanceError

1
2
3
4
5
6
7
8
9
10
11
12
cookiemon = User('cookiemon', 'mon@cookie.com', '111-111-1111', 'password')
session.add(cookiemon)
o1 = Order()
o1.user = cookiemon
session.add(o1)

cc = session.query(Cookie).filter(Cookie.cookie_name ==
"Change chocolate chip").one()
line1 = LineItem(order=o1, cookie=cc, quantity=2, extended_cost=1.00)

session.add(line1)
session.commit()
1
2
3
order = session.query(Order).first()
session.expunge(order)
order.line_items.all()

Transactions

1
from sql_orm_setup_cons import *

Defining the ship_it function:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
def ship_it(order_id):
order = session.query(Order).get(order_id)
for li in order.line_items:
li.cookie.quantity = li.cookie.quantity - li.quantity
session.add(li.cookie)
order.shipped = True
session.add(order)
session.commit()
print("shipped order ID: {}".format(order_id))

ship_it(1)
print(session.query(Cookie.cookie_name, Cookie.quantity).all())

ship_it(2)
print(session.query(Cookie.cookie_name, Cookie.quantity).all())
session.rollback()
print(session.query(Cookie.cookie_name, Cookie.quantity).all())

Transactional ship_it:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
from sqlalchemy.exc import IntegrityError
def ship_it(order_id):
order = session.query(Order).get(order_id)
for li in order.line_items:
li.cookie.quantity = li.cookie.quantity - li.quantity
session.add(li.cookie)
order.shipped = True
session.add(order)
try:
session.commit()
print("shipped order ID: {}".format(order_id))
except IntegrityError as error:
print('ERROR: {!s}'.format(error.orig))
session.rollback()

ship_it(2)

Testing with SQLAlchemy ORM

1
from sql_orm_setup_test import *

Testing with a Test Database

1
2
3
4
5
6
7
8
9
10
11
12
13
14
from db import Cookie, LineItem, Order, User,  dal


def get_orders_by_customer(cust_name, shipped=None, details=False):
query = dal.session.query(Order.order_id, User.username, User.phone)
query = query.join(User)
if details:
query = query.add_columns(Cookie.cookie_name, LineItem.quantity,
LineItem.extended_cost)
query = query.join(LineItem).join(Cookie)
if shipped is not None:
query = query.filter(Order.shipped == shipped)
results = query.filter(User.username == cust_name).all()
return results
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
import unittest

from decimal import Decimal

from db import prep_db, dal

from app import get_orders_by_customer


class TestApp(unittest.TestCase):
cookie_orders = [(1, u'cookiemon', u'111-111-1111')]
cookie_details = [
(1, u'cookiemon', u'111-111-1111',
u'dark chocolate chip', 2, Decimal('1.00')),
(1, u'cookiemon', u'111-111-1111',
u'oatmeal raisin', 12, Decimal('3.00'))]

@classmethod
def setUpClass(cls):
dal.conn_string = 'sqlite:///:memory:'
dal.connect()
dal.session = dal.Session()
prep_db(dal.session)
dal.session.close()

def setUp(self):
dal.session = dal.Session()

def tearDown(self):
dal.session.rollback()
dal.session.close()

def test_orders_by_customer_blank(self):
results = get_orders_by_customer('')
self.assertEqual(results, [])

def test_orders_by_customer_blank_shipped(self):
results = get_orders_by_customer('', True)
self.assertEqual(results, [])

def test_orders_by_customer_blank_notshipped(self):
results = get_orders_by_customer('', False)
self.assertEqual(results, [])

def test_orders_by_customer_blank_details(self):
results = get_orders_by_customer('', details=True)
self.assertEqual(results, [])

def test_orders_by_customer_blank_shipped_details(self):
results = get_orders_by_customer('', True, True)
self.assertEqual(results, [])

def test_orders_by_customer_blank_notshipped_details(self):
results = get_orders_by_customer('', False, True)
self.assertEqual(results, [])

def test_orders_by_customer_bad_cust(self):
results = get_orders_by_customer('bad name')
self.assertEqual(results, [])

def test_orders_by_customer_bad_cust_shipped(self):
results = get_orders_by_customer('bad name', True)
self.assertEqual(results, [])

def test_orders_by_customer_bad_cust_notshipped(self):
results = get_orders_by_customer('bad name', False)
self.assertEqual(results, [])

def test_orders_by_customer_bad_cust_details(self):
results = get_orders_by_customer('bad name', details=True)
self.assertEqual(results, [])

def test_orders_by_customer_bad_cust_shipped_details(self):
results = get_orders_by_customer('bad name', True, True)
self.assertEqual(results, [])

def test_orders_by_customer_bad_cust_notshipped_details(self):
results = get_orders_by_customer('bad name', False, True)
self.assertEqual(results, [])

def test_orders_by_customer(self):
results = get_orders_by_customer('cookiemon')
self.assertEqual(results, self.cookie_orders)

def test_orders_by_customer_shipped_only(self):
results = get_orders_by_customer('cookiemon', True)
self.assertEqual(results, [])

def test_orders_by_customer_unshipped_only(self):
results = get_orders_by_customer('cookiemon', False)
self.assertEqual(results, self.cookie_orders)

def test_orders_by_customer_with_details(self):
results = get_orders_by_customer('cookiemon', details=True)
self.assertEqual(results, self.cookie_details)

def test_orders_by_customer_shipped_only_with_details(self):
results = get_orders_by_customer('cookiemon', True, True)
self.assertEqual(results, [])

def test_orders_by_customer_unshipped_only_details(self):
results = get_orders_by_customer('cookiemon', False, True)
self.assertEqual(results, self.cookie_details)

Using Mocks

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
import unittest
from decimal import Decimal

import mock

from app import get_orders_by_customer


class TestApp(unittest.TestCase):
cookie_orders = [(1, u'cookiemon', u'111-111-1111')]
cookie_details = [
(1, u'cookiemon', u'111-111-1111',
u'dark chocolate chip', 2, Decimal('1.00')),
(1, u'cookiemon', u'111-111-1111',
u'oatmeal raisin', 12, Decimal('3.00'))]

@mock.patch('app.dal.session')
def test_orders_by_customer_blank(self, mock_dal):
mock_dal.query.return_value.join.return_value.filter.return_value. \
all.return_value = []
results = get_orders_by_customer('')
self.assertEqual(results, [])

@mock.patch('app.dal.session')
def test_orders_by_customer_blank_shipped(self, mock_dal):
mock_dal.query.return_value.join.return_value.filter.return_value. \
filter.return_value.all.return_value = []
results = get_orders_by_customer('', True)
self.assertEqual(results, [])

@mock.patch('app.dal.session')
def test_orders_by_customer(self, mock_dal):
mock_dal.query.return_value.join.return_value.filter.return_value. \
all.return_value = self.cookie_orders
results = get_orders_by_customer('cookiemon')
self.assertEqual(results, self.cookie_orders)

Reflection with SQLAlchemy ORM and Automap

Reflecting a Database with Automap

setup:

1
2
3
4
5
6
7
8
9
10
11
12
from sqlalchemy.ext.automap import automap_base
from sqlalchemy.orm import Session
from sqlalchemy import create_engine

Base = automap_base()

from sqlalchemy import create_engine

engine = create_engine('sqlite:///Chinook_Sqlite.sqlite')

Base.prepare(engine, reflect=True)
Base.classes.keys()
1
2
3
4
5
6
7
8
9
10
11
12
Artist = Base.classes.Artist
Album = Base.classes.Album

from sqlalchemy.orm import Session

session = Session(engine)
for artist in session.query(Artist).limit(10):
print(artist.ArtistId, artist.Name)

artist = session.query(Artist).first()
for album in artist.album_collection:
print('{} - {}'.format(artist.Name, album.Title))

Show password in web

1
document.querySelector("input[type=password]").value;

Show IP address

public:

1
curl ifconfig.me

private:

1
ipconfig

Download file and unzip

1
2
wget https://training.linuxfoundation.org/cm/LFS258/LFS258V2021-09-20SOLUTIONS.tar.xz --user=xxx --password=xxx
tar -xvf LFS258V2021-09-20SOLUTIONS.tar.xz

Send email from gmail

Reference:

Note:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
import smtplib
from email.mime.text import MIMEText
import ssl

port = 465
password = input("your password")
context = ssl.create_default_context()

msg = MIMEText("The body of the email is here")
msg['Subject'] = "An Email Alert"
msg['From'] = "my@gmail.com"
msg['To'] = "other@xxx.xxx"

with smtplib.SMTP_SSL("smtp.gmail.com", port, context=context) as server:
server.login("my@gmail.com", password)
server.send_message(msg)

Check the quality of a website

Setup personal PC as SSH server

Get started with OpenSSH for Windows

related commands:

1
2
New-NetFirewallRule -Name sshd -DisplayName 'OpenSSH Server (sshd)' -Enabled
True -Direction Inbound -Protocol TCP -Action Allow -Local Port 22
1
icacls.exe "C:\Users\<username>\.ssh\authorized_keys" /inheritance:r /grant "Administrators:F" /grant "SYSTEM:F"
1
Test-NetConnection -ComputerName 192.168.1.67 -Port 57859

reference

Example

  • email address: [A-Za-z0-9\._+]+@[A-Za-z]+\.(com|org|edu|net)

Merge Sort

Explanation:

  1. Wikipedia: https://en.wikipedia.org/wiki/Merge_sort
  2. Youtube: https://www.youtube.com/watch?time_continue=17&v=KF2j-9iSf4Q&feature=emb_logo

Time complexity: O(nlogn)
Space complexity: O(n)

Implementation:

Python

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47

def merge(arr1: list, arr2: list) -> tuple:
"""Merge two arrays

Args:
arr1 (list): first array
arr2 (list): second array

Returns:
tuple(int, list): number of swaps, sorted array
"""
result = []
count = 0
i, j = 0, 0
m, n = len(arr1), len(arr2)
while i < m and j < n:
if arr1[i] <= arr2[j]:
result.append(arr1[i])
i += 1
else:
result.append(arr2[j])
count += m - i
j += 1
result += arr1[i:]
result += arr2[j:]
return count, result


def msort(arr: list) -> tuple:
"""Merge Sort Algorithm

Args:
arr (list): unsorted array

Returns:
tuple(int, list): number of swaps, sorted array
"""
n = len(arr)
if n > 1:
mid = n // 2
left_swaps, left_result = msort(arr[:mid])
right_swaps, right_result = msort(arr[mid:])
merge_swaps, result = merge(left_result, right_result)
return left_swaps+right_swaps+merge_swaps, result
return 0, arr

assert msort([1, 2, 5, 6, 3, 7, 4, 8]) == (5, [1, 2, 3, 4, 5, 6, 7, 8])

LCS(longest common subsequence) problem

Explanation:

  1. Wikipedia: https://en.wikipedia.org/wiki/Longest_common_subsequence_problem

complexity: O(n × m)

Implementation

Python

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
def commonChild(s1: str, s2: str) -> int:
"""Find length of common child string

Args:
s1 (str): first string
s2 (str): second string

Returns:
int: length of common child string
"""
m = len(s1)
n = len(s2)
mat = [[0 for _ in range(n+1)] for _ in range(m+1)]
for r in range(1, m+1):
for c in range(1, n+1):
i, j = r - 1, c - 1
if s1[i] == s2[j]:
mat[r][c] = mat[r-1][c-1] + 1
else:
mat[r][c] = max(mat[r-1][c], mat[r][c-1])
return mat[m][n]

assert commonChild("SHINCHAN", "NOHARAAA") == 3
0%