# ── Packages ───────────────────────────────────────────────────────────────────
suppressPackageStartupMessages({
  library(arrow)
  library(dplyr)
  library(tidyr)
  library(ggplot2)
  library(kableExtra)
  library(scales)
  library(lubridate)
  library(yaml)
})

# ── Project helpers and paths ──────────────────────────────────────────────────
source("src/utils/helpers.R")
paths <- load_paths()

# Convenience: garch_shock_ext specification paths
sp <- paths$specs$garch_shock_ext

This notebook reproduces the full empirical analysis of “Who Stays Calm in Chaos? The Financial Instability of Crypto-Assets.” Each section integrates the paper text with the R code that produced it. Heavy estimations (DFM, GARCH, rolling QR) are wrapped in eval=FALSE blocks; the notebook loads results from pre-computed parquet files in output/models/.


Abstract

This article measures financial instability in crypto-asset markets using the quantile regression framework of Baur & Schulze (2009), extended to individual assets and estimated in 18-month rolling windows. We classify each asset as financially unstable or fragile based on whether its sensitivity to systematic shocks increases in the left or right tail of its return distribution. Applied to 137 crypto-assets over 2019–2026, approximately 10% of assets exhibit Financial Instability and 16% exhibit Financial Fragility on average. Financial Instability concentrates around the 2022 Crypto-Winter while Financial Fragility peaks during the 2021 bull market, consistent with the Minsky hypothesis. Crypto markets exhibit substantially higher instability and fragility than a benchmark of individual equity stocks. A predictive portfolio exercise confirms that stable assets generate higher cumulative returns than tail-amplifying assets over the full sample.

Keywords: Financial Stability · Crypto-Assets · Contagion · Systematic Risk · Quantile


1 Introduction

Crypto-assets have expanded rapidly in both popularity and market capitalization, leading to their gradual inclusion in investment portfolios. Yet, this growing adoption raises concerns about growing interactions with traditional markets, and the financial stability of crypto-markets is a major concern. Existing works fail to produce objective conclusions on the financial stability of crypto-markets. This paper addresses this gap by adopting an operational definition of financial asset stability from Baur & Schulze (2009) that yields a binary classification for each crypto-asset in each period.

While early adoption of crypto-assets remained limited to a niche group of enthusiasts, large holders (“whales”), and individuals engaged in illegal activities (Athey et al., 2016; Foley et al., 2019; Aiello et al., 2023), the creation of crypto-exchanges accelerated the development of crypto-markets since 2017. Several works document the dynamics of retail adoption (Lammer et al., 2020; Auer et al., 2022; Cornelli et al., 2023). On the institutional side, research consistently shows rising adoption (Huang et al., 2022; Auer & Ongena, 2022). This growing adoption raises concerns from regulators and academics regarding bubble dynamics (Panetta, 2022; Kyriazis et al., 2020) and the financial stability of crypto-markets more broadly (Grippa & Mann, 2022). However, existing measures do not provide a financial stability index grounded in a formal definition and independent of benchmark selection.

Financial market instability received greater attention in the aftermath of the Bretton-Woods system collapse (Minsky, 1977). The literature has converged on several complementary notions of financial stability. From a practical perspective, it represents the absence of financial instabilities or crises (Crockett, 1996; Chant, 2003). The Financial Instability Hypothesis (Minsky, 1977, 1992) argues that financial instability is endogenous: prolonged favorable conditions foster increasingly fragile financial structures. Schinasi (2004) defines financial stability as the capacity of a financial system to fulfil its functions resiliently. Allen & Gale (2008) argue that financial instabilities transform markets from shock absorbers to shock amplifiers. Against this backdrop, Baur & Schulze (2009) propose testing financial stability via quantile regressions.

The literature examining the financial stability of crypto-assets is extensive, yet only a few studies follow established definitions. Some articles use volatility comparisons (Bhosale & Mavale, 2018; Gupta et al., 2022; Baur & Dimpfl, 2021; Kristoufek, 2023), which remain sensitive to benchmark choice and are not a direct measure of financial stability. More recent works (Chatziantoniou et al., 2021; Ando et al., 2022; Bouri et al., 2021) introduce tail-sensitive connectedness measures, yet often rely on ad hoc criteria to label stability without enabling formal coefficient comparisons across quantiles.

This article proposes to fill this gap by employing the quantile regression framework of Baur & Schulze (2009) at the individual asset level for crypto-markets. Our three contributions are:

  1. Financial Instability as left-tail amplification. We propose a definition-grounded metric — the amplification of negative systematic shocks — rather than benchmark-dependent volatility measures. Approximately 10% of crypto-assets amplify systematic shocks in their extreme negative regimes on average.

  2. Asset-level, rolling-window application. Financial Instability is episodic, concentrating in three structurally distinct waves: the 2017–2018 Crypto-Winter, the Terra/LUNA–FTX sequence of 2022, and a third episode in late 2025.

  3. Financial Fragility index. Following the Financial Instability Hypothesis, we construct an index of upside amplification as an indicator of speculative build-up. Financial Fragility peaks during the 2021 bull market, consistent with Minsky (1977, 1992).

The rest of the article is organized as follows. Section 2 presents the data and methodology. Section 3 provides our results. Section 4 reports further analyses. Section 5 presents robustness checks. Section 6 concludes.


2 Methodology and Data

2.1 Data

We collect the list of the 40 most capitalized crypto-assets on the first week of each year from 2017 to 2025 from CoinMarketCap and remove stablecoins from the sample, leading to a final sample of 137 crypto-assets. Stablecoins are identified using CoinGecko and excluded under categories “Stablecoins,” “Fiat-backed Stablecoins,” “USD Stablecoins,” and “Crypto-backed Stablecoins.”

We extract daily price data from January 1, 2018 to February 1, 2026 from CryptoCompare. Returns are measured as log-price differences. To ensure liquidity, we remove observations when the volume is below 10 tokens.1

# ── Load the final crypto panel ────────────────────────────────────────────────
# data_crypto.parquet contains log returns, market caps, and — crucially —
# resid_garch_std: the GARCH-standardised systematic shock z_t = ε_t/σ_t.
data_crypto <- arrow::read_parquet(paths$data$final$data_crypto) %>%
  mutate(date = as.Date(date)) %>%
  select(date, variable, return, resid_garch_std)

cat("Panel:", nrow(data_crypto), "obs —",
    length(unique(data_crypto$variable)), "tokens —",
    format(min(data_crypto$date)), "to", format(max(data_crypto$date)), "\n")
Panel: 312623 obs — 145 tokens — 2018-01-01 to 2026-02-01 
# ── Summary statistics ─────────────────────────────────────────────────────────
make_stats_row <- function(x, label, n_assets = NA_character_) {
  data.frame(
    Series       = label,
    Mean         = round(mean(x, na.rm=TRUE), 3),
    `Std. Dev.`  = round(sd(x, na.rm=TRUE), 3),
    Median       = round(median(x, na.rm=TRUE), 3),
    Minimum      = round(min(x, na.rm=TRUE), 3),
    Maximum      = round(max(x, na.rm=TRUE), 3),
    Observations = formatC(sum(!is.na(x)), format="d", big.mark=","),
    Assets       = n_assets,
    check.names  = FALSE
  )
}

ret_vec   <- data_crypto$return[!is.na(data_crypto$return)]
shock_vec <- data_crypto %>%
  filter(!is.na(resid_garch_std)) %>%
  distinct(date, resid_garch_std) %>%
  pull(resid_garch_std)

bind_rows(
  make_stats_row(ret_vec,   "Log returns",      n_assets = as.character(n_distinct(data_crypto$variable))),
  make_stats_row(shock_vec, "Systematic shock",  n_assets = "—")
) %>%
  kable(caption = "**Table 1 — Summary Statistics**",
        align = c("l","r","r","r","r","r","r","r")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 13) %>%
  footnote(
    general = "Log returns are daily log price changes for 137 crypto-assets over January 2018 to February 2026. The systematic shock is the standardised residual z_t = ε_t/σ_t of an eGARCH(1,1) model applied to the first-differenced Crypto-Factor.",
    general_title = "Note: "
  )
**Table 1 — Summary Statistics**
Series Mean Std. Dev. Median Minimum Maximum Observations Assets
Log returns -0.001 0.139 -0.001 -22.928 22.924 312,478 145
Systematic shock 0.007 0.969 0.032 -8.215 6.939 2,954
Note:
Log returns are daily log price changes for 137 crypto-assets over January 2018 to February 2026. The systematic shock is the standardised residual z_t = ε_t/σ_t of an eGARCH(1,1) model applied to the first-differenced Crypto-Factor.

2.2 Systematic Shocks

To estimate systematic shocks in crypto-markets, we specify a Dynamic Factor Model (DFM) for a panel of crypto-asset prices. As Rey & Miranda-Agrippino (2021) argue, the single factor of equity prices combines market risk and aggregate risk-aversion. We maintain this interpretation for crypto markets.

Let pt denote an n-dimensional vector of daily prices pi, t that are a linear combination of a common factor ft following an AR(p) process and an idiosyncratic error term εi, t:

pi, t = λi(L) ft + εi, t

ft = A1ft − 1 + ⋯ + Aqft − q + ηt

where ηt ∼ 𝒩(0,Σ) and λi is a vector of factor loadings for asset i. Following Che et al. (2023), we estimate the single factor on the 7 biggest crypto-assets by market capitalization (Bitcoin, Ethereum, Ripple, Binance Coin, Cardano, TRON and DOGE), representing around 70% of the crypto-market capitalization at the end of our sample.

# ── DFM estimation (eval=FALSE — run once via src/02_factor_and_shocks.R) ─────
# Inputs:  data/raw/* price data from CryptoCompare
# Outputs: data/intermediate/market_shocks.parquet  (contains the DFM factor
#          series, raw GARCH residuals, and resid_garch_std_crypto = z_t)
#
# The eGARCH(1,1) with Student distribution is then applied in
# src/02c_garch_std_shock.R, which patches data_crypto.parquet to add
# resid_garch_std = z_t = ε_t / σ_t.
source("src/02_factor_and_shocks.R")
source("src/02c_garch_std_shock.R")

We construct the systematic shock ft* as the standardised residuals of an eGARCH(1,1) model2 applied to the Crypto-Factor in first difference for stationarity:

$$f^*_t = z_t = \frac{\varepsilon_t}{\sigma_t}$$

where εt is the GARCH residual and σt is the estimated conditional standard deviation. By construction, zt has unit scale at every date, making cross-window shock distributions comparable. This matters because the GARCH residuals exhibit substantial variation in conditional variance across subsamples: a unit shock in a calm window is not comparable to a unit shock in a turbulent window. Standardising by σt removes this scale heterogeneity while preserving the variance-transmission channel in the dependent variable — a key distinction from alternative scalings that filter the returns themselves.3

# z_t is stored directly in data_crypto.parquet (column resid_garch_std)
# It is also in data/intermediate/market_shocks.parquet as resid_garch_std_crypto

shock_series <- data_crypto %>%
  distinct(date, z_t = resid_garch_std) %>%
  filter(!is.na(z_t)) %>%
  arrange(date)

ggplot(shock_series, aes(x = date, y = z_t)) +
  geom_line(color = "black", linewidth = 0.4, alpha = 0.85) +
  geom_hline(yintercept = 0, linetype = "dashed", color = "grey50") +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.01)) +
  labs(x = NULL, y = expression(z[t] == epsilon[t] / sigma[t])) +
  theme_crypto()

# The factor series is saved in market_shocks.parquet
market_shocks <- arrow::read_parquet(paths$data$intermediate$market_shocks) %>%
  mutate(date = as.Date(date))

ggplot(market_shocks, aes(x = date, y = serie)) +
  geom_line(color = "black", linewidth = 0.6) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.01)) +
  labs(x = NULL, y = "Factor level") +
  theme_crypto()

2.3 Quantile Regression Framework

Our empirical strategy builds on Baur & Schulze (2009), who define financial asset stability as the absence of increasing shock sensitivity in the tails of the return distribution relative to normal conditions. For each asset i, we estimate the following quantile regression at three quantiles τ — the lower tail (τ = 0.05), the median (τ = 0.50), and the upper tail (τ = 0.95):

Qτ(ritft*) = αi(τ) + βi(τ) ft* + γi(τ) ft* ⋅ Dtlower + ϕi(τ) ft* ⋅ Dtupper + εi(τ)

where rit is the return of asset i, ft* = zt is the standardised systematic shock, and Dtlower (Dtupper) is a dummy equal to one when ft* falls in the bottom (top) 5% of its in-window distribution.

The bilateral shock dummies isolate the relationship between asset return regimes and systematic shocks of ordinary magnitude. Without them, βi(τ) would conflate the asset’s intrinsic regime-dependent sensitivity with the general co-movement induced by tail market events — confounding tail-dependence with co-exceedances. Under this setup, βi(τ) identifies how asset return regimes relate to the systematic factor under normal market conditions, while γi and ϕi capture additional sensitivity under extreme shocks.

The equation delivers three quantile-specific estimates: β̂lower ≡ β̂i(0.05), β̂med ≡ β̂i(0.50), and β̂upper ≡ β̂i(0.95). Each tail is tested against the median via a pairwise Wald test (Koenker & Machado, 1999) at the 10% significance level.

Classification rules:

  • Financial Instability (left-tail amplification): β̂lower > β̂med and Wald test rejects at 10%. Shock transmission intensifies when assets are in distress, amplifying aggregate downside risk (Forbes & Rigobon, 2001; Longin & Solnik, 2001).

  • Financial Fragility (right-tail amplification): β̂upper > β̂med and Wald test rejects at 10%. Assets amplifying positive market shocks during expansionary phases fuel price bubbles and create correlated exposures that subsequently unwind (Brunnermeier et al., 2013), consistent with the Minsky hypothesis.

To capture time variation, we estimate the equation within 18-month rolling windows, stepping monthly. Assets are included only if their return coverage exceeds 90% of the available trading days in a given window.4 Each window is self-contained: all estimates and shock quantile thresholds are computed within the window, preventing contamination by out-of-window information.

# ── Rolling QR estimation (eval=FALSE — run once via src/25_garch_shock_full.R)
#
# The function run_rolling_qr_analysis() in helpers.R parallelises over windows.
# It reads data_crypto.parquet (which already contains resid_garch_std = z_t),
# and applies the formula:
#   return ~ resid_garch_std + resid_garch_std:d_low + resid_garch_std:d_high
# with bilateral dummies (add_dummies = "bilateral", shock_q = 0.05).
#
# Output: output/models/roll_18_garch_shock_ext.parquet
#   Columns: end, variable, left_tail, right_tail,
#            beta_lower, beta_median, beta_upper,
#            p_lower, p_upper, instable, cont, spec
source("src/25_garch_shock_full.R")
# ── Load pre-computed rolling QR results ──────────────────────────────────────
roll <- arrow::read_parquet(sp$models$roll_18) %>%
  mutate(end = as.Date(end))

cat("Rolling results:", nrow(roll), "rows —",
    length(unique(roll$variable)), "tokens —",
    length(unique(roll$end)), "windows\n")
Rolling results: 7760 rows — 137 tokens — 80 windows
cat("Classification columns: cont (FI), spec (FF), instable\n")
Classification columns: cont (FI), spec (FF), instable
roll %>%
  group_by(end) %>%
  summarise(n = n_distinct(variable), .groups = "drop") %>%
  ggplot(aes(x = end, y = n)) +
  geom_line(color = "black", linewidth = 0.7) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.01)) +
  scale_y_continuous(limits = c(0, NA), expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Number of assets") +
  theme_crypto()


3 Results

3.1 Financial Instability and Financial Fragility Dynamics

Figure 1 reports the results of our classification strategy in rolling windows, alongside crypto market capitalisation. On average, approximately 10% of crypto-assets exhibit left-tail amplification of systematic shocks. The fraction exhibiting Financial Fragility is higher, averaging 15%.

# ── Rolling shares ─────────────────────────────────────────────────────────────
shares <- roll %>%
  group_by(end) %>%
  summarise(
    fi   = mean(cont, na.rm = TRUE) * 100,
    ff   = mean(spec, na.rm = TRUE) * 100,
    n    = n_distinct(variable),
    .groups = "drop"
  )

shares %>%
  summarise(
    `FI mean (%)`  = round(mean(fi), 1),
    `FI max (%)`   = round(max(fi), 1),
    `FF mean (%)`  = round(mean(ff), 1),
    `FF max (%)`   = round(max(ff), 1),
    `Avg. N`       = round(mean(n))
  ) %>%
  kable(caption = "**Average FI/FF shares across 80 rolling windows**") %>%
  kable_styling(bootstrap_options = c("striped","hover"), full_width = FALSE)
**Average FI/FF shares across 80 rolling windows**
FI mean (%) FI max (%) FF mean (%) FF max (%) Avg. N
10.1 33.3 15.8 35.6 97
shares %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(
    recode(metric, fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")
  )) %>%
  ggplot(aes(x = end, y = share / 100, color = metric, linetype = metric)) +
  geom_line(linewidth = 1) +
  scale_color_manual(
    values = c("Financial Instability" = "#d62728",
               "Financial Fragility"   = "#1f77b4"),
    name = NULL) +
  scale_linetype_manual(
    values = c("Financial Instability" = "solid",
               "Financial Fragility"   = "solid"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

The Financial Instability share is episodic rather than persistent. Three structurally distinct episodes emerge:

  1. 2019: tail of the 2017–2018 Crypto-Winter, following Bitcoin’s peak near $20,000 in December 2017 and a subsequent decline exceeding 80%.

  2. Mid-2022 (peak: November 2022): the Terra/LUNA implosion in May 2022, followed by the failures of Three Arrows Capital and Celsius Network, and the FTX exchange collapse in November 2022. A secondary elevation persists into early 2023, coinciding with the Silicon Valley Bank failure and the brief USDC depeg.

  3. Late 2025: a third episode at the end of the estimation sample.

The Financial Fragility share exhibits a different dynamic: elevated and relatively persistent over 2020–2024. It peaks above 30% in mid-2021 during the bull market following the initial Covid-19 recovery, consistent with Lahmiri & Bekiros (2020), and remains above its sample mean through 2024. This pattern is consistent with Minsky’s Financial Instability Hypothesis (1992): speculative financing and herding accumulate during sustained price appreciation, manifesting as right-tail amplification. The increase in Financial Fragility slightly precedes the market capitalisation peak, consistent with behavioural amplification preceding aggregate overvaluation (Brunnermeier & Nagel, 2009; Stolborg et al., 2025).

Neither index co-moves systematically with aggregate market volatility over the full sample. For Financial Instability, the absence of a positive correlation with volatility is consistent with studies arguing that volatility is not a good predictor of financial crises (Danielsson et al., 2018; Sornette & Cauwels, 2018) and justifies our approach.

Financial Fragility exceeds Financial Instability throughout the sample, reflecting the asymmetric structure of crypto market cycles. Boom phases are prolonged and broad-based relative to sharp but brief crash episodes: during expansionary phases, speculative demand and momentum trading cause a large cross-section of assets to amplify positive systematic shocks simultaneously (Minsky, 1977; Bouri et al., 2019). Crash episodes, by contrast, are concentrated in time and the amplification of negative shocks is more selective. The result is a persistently higher share of assets exhibiting right-tail amplification than left-tail amplification.


4 Further Analysis

4.1 Portfolio Performance

We construct four equally-weighted portfolios formed on the rolling classification, rebalanced monthly with a one-month forward lag to ensure the classification is fully out-of-sample:

  • Financial Instability portfolio: holds assets with β̂i(0.05) > β̂i(0.50) exclusively
  • Financial Fragility portfolio: holds assets with β̂i(0.95) > β̂i(0.50) exclusively
  • Stable portfolio: all remaining assets satisfying neither criterion
  • Top-10 benchmark: equally-weighted portfolio of the ten most prominent crypto-assets
# ── Portfolio construction (eval=FALSE — run once via src/12_portfolio_performance.R)
# Uses roll_18_garch_shock_ext.parquet + daily return data from data_crypto.parquet.
# Classification at window end t → portfolio held over month t+1.
# Output: output/figures/garch_shock_ext/paper/portfolio_predictive.pdf
source("src/12_portfolio_performance.R")

# ── Helper functions (from src/12_portfolio_performance.R) ────────────────────
build_constituents <- function(roll_df, portfolio_type) {
  if (portfolio_type == "bench") {
    return(roll_df %>% distinct(end, variable) %>% mutate(in_portfolio = TRUE))
  }
  roll_df %>%
    distinct(end, variable, instable, cont, spec) %>%
    mutate(in_portfolio = case_when(
      portfolio_type == "cont_only" ~ (as.logical(cont) & !as.logical(spec)),
      portfolio_type == "spec_only" ~ (as.logical(spec) & !as.logical(cont)),
      portfolio_type == "stable"    ~ (!as.logical(instable) & !is.na(instable)),
      TRUE ~ FALSE
    )) %>%
    select(end, variable, in_portfolio)
}

build_constituents_lagged <- function(roll_df, portfolio_type) {
  build_constituents(roll_df, portfolio_type) %>%
    arrange(variable, end) %>%
    group_by(variable) %>%
    mutate(in_portfolio = lag(in_portfolio, default = FALSE)) %>%
    ungroup()
}

compute_daily_returns <- function(daily_df, constituents_df) {
  dates_sched <- sort(unique(constituents_df$end))
  ret_cap     <- log(2)
  purrr::map_dfr(seq_along(dates_sched), function(i) {
    t_start <- dates_sched[i]
    t_end   <- if (i < length(dates_sched)) dates_sched[i + 1L] else max(daily_df$date)
    members <- constituents_df %>% filter(end == t_start, in_portfolio) %>% pull(variable)
    rows    <- daily_df %>% filter(date > t_start, date <= t_end)
    if (length(members) == 0)
      return(rows %>% distinct(date) %>% mutate(port_ret = NA_real_, n_assets = 0L))
    wts <- tibble(variable = members, weight = 1 / length(members))
    rows %>%
      filter(variable %in% members) %>%
      left_join(wts, by = "variable") %>%
      mutate(simple_ret = exp(pmax(pmin(return, ret_cap), -ret_cap)) - 1) %>%
      group_by(date) %>%
      summarise(port_ret  = sum(simple_ret * weight, na.rm = TRUE) /
                              max(sum(weight[!is.na(simple_ret)]), 1e-9),
                n_assets  = sum(!is.na(simple_ret)),
                .groups = "drop")
  })
}

# ── Top-10 fixed basket ────────────────────────────────────────────────────────
TOP10 <- c("BTC","ETH","BNB","XRP","ADA","SOL","DOGE","TRX","LTC","DOT")
top10_assets <- intersect(TOP10, unique(roll$variable))
top10_const  <- roll %>% distinct(end) %>%
  crossing(variable = top10_assets) %>%
  mutate(in_portfolio = TRUE)

# ── Build lagged predictive portfolios ────────────────────────────────────────
port_types <- c(cont_only = "Financial Instability",
                spec_only = "Financial Fragility",
                stable    = "Stable")

port_data <- purrr::imap_dfr(port_types, function(label, ptype) {
  build_constituents_lagged(roll, ptype) %>%
    compute_daily_returns(data_crypto, .) %>%
    mutate(portfolio = label)
})

top10_data <- compute_daily_returns(data_crypto, top10_const) %>%
  mutate(portfolio = "Top-10")

# ── Cumulative wealth ──────────────────────────────────────────────────────────
PORT_LEVELS <- c("Stable", "Top-10", "Financial Fragility", "Financial Instability")
COLORS_PORT <- c("Financial Instability" = "#d62728",
                 "Financial Fragility"   = "#1f77b4",
                 "Stable"                = "darkgreen",
                 "Top-10"                = "black")

bind_rows(port_data, top10_data) %>%
  mutate(portfolio = factor(portfolio, levels = PORT_LEVELS)) %>%
  group_by(portfolio) %>%
  arrange(date) %>%
  mutate(cum_wealth = 100 * cumprod(ifelse(is.na(port_ret), 1, 1 + port_ret))) %>%
  ungroup() %>%
  ggplot(aes(x = date, y = cum_wealth, color = portfolio, linetype = portfolio)) +
  geom_line(linewidth = 0.9, na.rm = TRUE) +
  scale_y_log10(labels = scales::label_number(accuracy = 1)) +
  scale_color_manual(values = COLORS_PORT, name = NULL) +
  scale_linetype_manual(
    values = c("Financial Instability" = "solid", "Financial Fragility" = "solid",
               "Stable" = "solid", "Top-10" = "solid"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  labs(x = NULL, y = "Cumulative wealth (log scale, base 100)") +
  theme_crypto()

The Stable portfolio generates the highest cumulative return over the full sample, outperforming both the Top-10 benchmark and the tail-amplification portfolios. Assets that do not amplify systematic shocks in either tail are less exposed to sharp drawdowns during stress episodes while still participating in broad market appreciation during calm phases, consistent with Ang et al. (2006). The result is consistent with the long-term performance advantage of assets avoiding crash amplification or bubble formation (Pellegrini et al., 2008).

The Financial Instability and Financial Fragility portfolios both underperform over the full sample. Both portfolios expand sharply during the 2021 bull market but decouple from the benchmark during the 2022 Crypto-Winter. This reflects the twin costs of tail amplification: assets exhibiting left-tail sensitivity underperform during downturns through shock amplification, while assets with right-tail sensitivity may overperform short-term during speculative phases but experience reversals once the speculative phase ends.

Overall, our tail-dependence portfolio analysis demonstrates the importance of incorporating tail-dependence in portfolio strategies, consistent with Bergmann et al. (2018) who show that tail dependence models outperform the Markowitz model in terms of cumulative return.

4.2 Comparisons with Equity Markets

To benchmark crypto instability against a traditional asset class, we apply the identical rolling quantile regression framework to individual equity stocks. Adjusted daily closing prices are sourced from Yahoo Finance for a universe of 143 large-cap firms spanning North America (60 stocks), Europe (54 stocks), Japan (13 stocks), China, Australia, and India (4 stocks).

The equity sample begins in January 1995 and closes in February 2026. The systematic equity shock is constructed by applying the same two-step procedure: a first principal component is extracted from 49 country-level equity price indices sourced from Datastream following Rey & Miranda-Agrippino (2021), and a GARCH(1,1) model is estimated on the first-differenced factor. Individual stocks are not used in factor construction, so the systematic shock is exogenous by design.

# ── Equity rolling QR (eval=FALSE — run once via src/25_garch_shock_full.R)
# Uses data/intermediate/stocks_panel.parquet.
# Output: output/models/stocks_roll_garch_shock_ext.parquet
#         output/figures/garch_shock_ext/equity/stocks_comparison.pdf
equity <- arrow::read_parquet(sp$models$stocks_roll) %>%
  mutate(end = as.Date(end))

cat("Equity results:", nrow(equity), "rows —",
    length(unique(equity$variable)), "stocks\n")
Equity results: 10804 rows — 143 stocks
bind_rows(
  data.frame(Group = "Crypto",
             `FI mean (%)` = round(mean(shares$fi), 1),
             `FF mean (%)` = round(mean(shares$ff), 1)),
  equity %>%
    group_by(end) %>%
    summarise(fi = mean(cont, na.rm=TRUE)*100,
              ff = mean(spec, na.rm=TRUE)*100, .groups="drop") %>%
    summarise(Group = "Equity",
              `FI mean (%)` = round(mean(fi), 1),
              `FF mean (%)` = round(mean(ff), 1))
) %>%
  kable(caption = "**Average Financial Instability and Fragility: Crypto vs. Equity**") %>%
  kable_styling(bootstrap_options = c("striped","hover"), full_width = FALSE)
**Average Financial Instability and Fragility: Crypto vs. Equity**
Group FI.mean.... FF.mean.... FI mean (%) FF mean (%)
Crypto 10.1 15.8 NA NA
Equity NA NA 5.3 4.6
equity_shares <- equity %>%
  group_by(end) %>%
  summarise(fi = mean(cont, na.rm=TRUE)*100,
            ff = mean(spec, na.rm=TRUE)*100, .groups="drop")

bind_rows(
  shares      %>% select(end, fi, ff) %>% mutate(group = "Crypto"),
  equity_shares                        %>% mutate(group = "Equity")
) %>%
  filter(end >= as.Date("2019-07-01")) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
                           fi = "Financial Instability",
                           ff = "Financial Fragility"),
                    levels = c("Financial Instability", "Financial Fragility")),
    group  = factor(group, levels = c("Crypto", "Equity"))
  ) %>%
  ggplot(aes(x = end, y = share / 100, color = group, linetype = group)) +
  geom_line(linewidth = 0.9, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(values = c("Crypto" = "black", "Equity" = "#1f77b4"),
                     name = NULL) +
  scale_linetype_manual(values = c("Crypto" = "solid", "Equity" = "dashed"),
                        name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

Equity markets exhibit substantially lower Financial Instability than crypto-assets, with an average share of 5.3% against 10.1% for crypto. A pronounced peak emerges in the immediate aftermath of the Covid-19 shock in 2020, during which synchronised global equity sell-offs generated left-tail amplification across a broad cross-section of individual stocks (Ramelli & Wagner, 2020; Alfaro et al., 2020). Outside of this episode, Financial Instability in equity markets remains well below the crypto benchmark.

The gap is even more pronounced for Financial Fragility. The average equity share of 4.6% is less than one-third of the crypto average of 15.7%. This divergence reflects the greater susceptibility of crypto markets to herding and momentum during expansionary phases (Bouri et al., 2021; Kyriazis et al., 2020): whereas equity prices respond to fundamental revisions in cash-flow expectations during booms, crypto valuations are more directly driven by speculative demand and sentiment. In addition, equity markets exhibit larger instability than fragility, consistent with Hu (2006) who shows that stock markets are more likely to exhibit left-tail dependence than right-tail dependence.

Taken together, these results establish a quantitative gap in financial stability between crypto-assets and equity markets. The novelty lies not in the direction of the finding but in its grounding: the Financial Instability and Financial Fragility indices are derived from a formal test of shock transmission stability rather than from a comparison against an arbitrary benchmark — a distinction that matters because volatility-based rankings are sensitive to the choice of reference asset, whereas our classification is self-contained at the asset level.


5 Robustness

5.1 Model Specification

We run several robustness checks to confirm that our results are not dependent on model specifications. The baseline uses 18-month rolling windows, τ ∈ {0.05, 0.50, 0.95}, Wald test at 10% significance, and bilateral dummies at the 5% tail threshold.

# ── All robustness estimations (eval=FALSE — run once via src/robust_garch_shock_ext.R)
# Outputs:
#   output/models/garch_shock_ext/robust_windows.parquet
#   output/models/garch_shock_ext/robust_tau_vec.parquet
#   output/models/garch_shock_ext/robust_alpha.parquet
#   output/models/garch_shock_ext/robust_threshold.parquet
# and corresponding PDF figures in output/figures/garch_shock_ext/robustness/
source("src/robust_garch_shock_ext.R")

5.1.1 Alternative Window Widths

We use shorter (6 and 12 months) and longer (24 and 30 months) windows. Shorter windows are more reactive but noisier; longer windows are smoother but slower to detect crises.

rob_win <- arrow::read_parquet(sp$models$robust_windows) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, window_months) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(recode(metric,
    fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")))

band_win  <- rob_win %>%
  group_by(end, metric) %>%
  summarise(lo = min(share), hi = max(share), .groups = "drop")
base_win  <- rob_win %>% filter(window_months == 18)

ggplot(band_win, aes(x = end)) +
  geom_ribbon(aes(ymin = lo, ymax = hi), fill = "grey70", alpha = 0.5) +
  geom_line(data = base_win, aes(y = share), color = "black", linewidth = 0.9) +
  facet_wrap(~metric, ncol = 2) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

Despite the heterogeneity introduced by window choice, our results stay consistent across specifications — particularly for the Financial Instability index, while the Financial Fragility index exhibits larger timing differences but a similar overall dynamics.

5.1.2 Alternative Quantile Vectors

We use a stricter specification (τ ∈ {0.01, 0.50, 0.99}) and a milder specification (τ ∈ {0.10, 0.50, 0.90}).

rob_tau <- arrow::read_parquet(sp$models$robust_tau_vec) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, tau_spec) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(recode(metric,
    fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")))

band_tau <- rob_tau %>%
  group_by(end, metric) %>%
  summarise(lo = min(share), hi = max(share), .groups = "drop")
base_tau <- rob_tau %>% filter(tau_spec == "baseline")

ggplot(band_tau, aes(x = end)) +
  geom_ribbon(aes(ymin = lo, ymax = hi), fill = "grey70", alpha = 0.5) +
  geom_line(data = base_tau, aes(y = share), color = "black", linewidth = 0.9) +
  facet_wrap(~metric, ncol = 2) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

The Financial Instability index is robust to the change of quantiles. The Financial Fragility index exhibits larger variation under the stricter specification, but the dynamics remains similar overall.

5.1.3 Alternative Significance Levels

We report results with a tighter 5% Wald test significance level alongside the baseline 10%.

rob_alpha <- arrow::read_parquet(sp$models$robust_alpha) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, alpha_val) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  filter(abs(alpha_val - 0.05) < 1e-9 | abs(alpha_val - 0.10) < 1e-9) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
      fi = "Financial Instability", ff = "Financial Fragility"),
      levels = c("Financial Instability", "Financial Fragility")),
    alpha_label = factor(paste0(alpha_val * 100, "%"), levels = c("5%", "10%"))
  )

ggplot(rob_alpha, aes(x = end, y = share,
                      color = alpha_label, linetype = alpha_label)) +
  geom_line(linewidth = 0.9, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(values = c("5%" = "grey60", "10%" = "black"), name = NULL) +
  scale_linetype_manual(values = c("5%" = "solid", "10%" = "solid"), name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

Consistent with using a tighter threshold, the level of Financial Instability and Financial Fragility decrease, yet the dynamics of both indices do not substantially change.

5.1.4 Alternative Extreme-Shock Thresholds

We report results for q ∈ {2.5%, 7.5%, 10%} against the baseline q = 5%.

rob_thr <- arrow::read_parquet(sp$models$robust_threshold) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, q_threshold) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(recode(metric,
    fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")))

band_thr <- rob_thr %>%
  group_by(end, metric) %>%
  summarise(lo = min(share), hi = max(share), .groups = "drop")
base_thr <- rob_thr %>% filter(abs(q_threshold - 0.05) < 1e-9)

ggplot(band_thr, aes(x = end)) +
  geom_ribbon(aes(ymin = lo, ymax = hi), fill = "grey70", alpha = 0.5) +
  geom_line(data = base_thr, aes(y = share), color = "black", linewidth = 0.9) +
  facet_wrap(~metric, ncol = 2) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

These alternative specifications confirm that the construction of Financial Instability and Financial Fragility is not dependent on the definition of extreme systematic shock dummies.

5.2 Selection in the Asset Universe

A potential concern is that large-cap assets may be structurally more stable than smaller tokens (Amihud, 2002), so that restricting the sample to assets ever ranked in the top 50 could understate Financial Instability and Fragility in the broader crypto universe. To assess this, we replicate the rolling estimation on an independent sample of 150 tokens listed on CoinMarketCap before January 2019 but absent from the baseline universe.

# ── Extended universe estimation (eval=FALSE — run once via src/tmp_extended_universe.R)
# Downloads 150 tokens via crypto2 (CoinMarketCap, no API key needed).
# Output: output/models/roll_extended_garch_shock_ext.parquet
#         output/figures/garch_shock_ext/robustness/selection_bias_extended.pdf
source("src/tmp_extended_universe.R")
# ── Selection bias summary table ───────────────────────────────────────────────
roll_ext <- arrow::read_parquet(
  "output/models/roll_extended_garch_shock_ext.parquet"
) %>% mutate(end = as.Date(end))

ext_shares <- roll_ext %>%
  group_by(end) %>%
  summarise(fi = mean(cont, na.rm=TRUE)*100,
            ff = mean(spec, na.rm=TRUE)*100, .groups="drop")

data.frame(
  Index   = rep(c("Financial Instability","Financial Fragility"), 2),
  Universe = c(rep("Baseline (top-50, N=137)", 2), rep("Extended (smaller tokens, N≈146)", 2)),
  Mean    = c(round(mean(shares$fi),1), round(mean(shares$ff),1),
              round(mean(ext_shares$fi),1), round(mean(ext_shares$ff),1)),
  SD      = c(round(sd(shares$fi),1),   round(sd(shares$ff),1),
              round(sd(ext_shares$fi),1), round(sd(ext_shares$ff),1)),
  Min     = c(round(min(shares$fi),1),  round(min(shares$ff),1),
              round(min(ext_shares$fi),1), round(min(ext_shares$ff),1)),
  Max     = c(round(max(shares$fi),1),  round(max(shares$ff),1),
              round(max(ext_shares$fi),1), round(max(ext_shares$ff),1))
) %>%
  arrange(Index, Universe) %>%
  kable(caption = "**Table 2 — Financial Instability and Fragility: Baseline vs. Extended Universe**",
        col.names = c("Index","Universe","Mean (%)","Std. Dev.","Min","Max"),
        align = c("l","l","r","r","r","r")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 13) %>%
  footnote(general = "Time-series statistics computed over rolling estimation windows. Extended universe: 150 tokens listed before January 2019 and absent from the baseline sample, downloaded via CoinMarketCap (crypto2 package).")
**Table 2 — Financial Instability and Fragility: Baseline vs. Extended Universe**
Index Universe Mean (%) Std. Dev. Min Max
Financial Fragility Baseline (top-50, N=137) 15.8 6.6 4.6 35.6
Financial Fragility Extended (smaller tokens, N≈146) 22.0 17.1 1.4 58.4
Financial Instability Baseline (top-50, N=137) 10.1 6.4 1.0 33.3
Financial Instability Extended (smaller tokens, N≈146) 9.7 12.7 0.0 64.4
Note:
Time-series statistics computed over rolling estimation windows. Extended universe: 150 tokens listed before January 2019 and absent from the baseline sample, downloaded via CoinMarketCap (crypto2 package).

The Financial Instability index is robust: the average share of 9.7% in the extended universe is nearly identical to the baseline 10.1%, and the episodic dynamics are preserved. The Financial Fragility share is systematically higher among smaller tokens (22.0% versus 15.8%), reflecting their greater susceptibility to speculative amplification during boom phases — a genuine compositional feature rather than a downward bias in the baseline.

[WARNING] Could not convert TeX math f^*_t = z_t = \frac{\varepsilon_t}{\sigma_t}, rendering as TeX
bind_rows(
  shares     %>% select(end, fi, ff) %>% mutate(group = "Top-50 universe (baseline)"),
  ext_shares %>% select(end, fi, ff) %>% mutate(group = "Extended universe (smaller tokens)")
) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
      fi = "Financial Instability", ff = "Financial Fragility"),
      levels = c("Financial Instability", "Financial Fragility")),
    group  = factor(group, levels = c("Top-50 universe (baseline)",
                                      "Extended universe (smaller tokens)"))
  ) %>%
  ggplot(aes(x = end, y = share / 100, color = group, linetype = group)) +
  geom_line(linewidth = 0.85, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(
    values = c("Top-50 universe (baseline)" = "black",
               "Extended universe (smaller tokens)" = "#d62728"),
    name = NULL) +
  scale_linetype_manual(
    values = c("Top-50 universe (baseline)" = "solid",
               "Extended universe (smaller tokens)" = "dashed"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

5.3 Endogeneity of Factor Constituents

The systematic shock ft* is extracted from a DFM estimated on seven benchmark assets. Because these seven assets contribute directly to the construction of the shock, their estimated quantile betas may be mechanically inflated. To assess this, we recompute the shares after excluding the seven factor constituents from the classification. No re-estimation is required: we simply exclude BTC, ETH, XRP, BNB, ADA, TRX, and DOGE when aggregating the shares.

# ── Factor constituent exclusion (eval=FALSE — run once via src/tmp_selection_bias_basket.R)
# Output: output/figures/garch_shock_ext/robustness/selection_bias.pdf
source("src/tmp_selection_bias_basket.R")
BASKET <- c("BTC", "ETH", "XRP", "BNB", "ADA", "TRX", "DOGE")

restr_shares <- roll %>%
  filter(!(variable %in% BASKET)) %>%
  group_by(end) %>%
  summarise(fi = mean(cont, na.rm=TRUE)*100,
            ff = mean(spec, na.rm=TRUE)*100, .groups="drop")

bind_rows(
  shares       %>% select(end, fi, ff) %>% mutate(group = "Baseline"),
  restr_shares %>% select(end, fi, ff) %>% mutate(group = "Excl. factor constituents")
) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
      fi = "Financial Instability", ff = "Financial Fragility"),
      levels = c("Financial Instability", "Financial Fragility")),
    group  = factor(group, levels = c("Baseline", "Excl. factor constituents"))
  ) %>%
  ggplot(aes(x = end, y = share / 100, color = group, linetype = group)) +
  geom_line(linewidth = 0.85, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(
    values = c("Baseline" = "black", "Excl. factor constituents" = "#d62728"),
    name = NULL) +
  scale_linetype_manual(
    values = c("Baseline" = "solid", "Excl. factor constituents" = "dashed"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()

The results confirm that the factor constituents do not substantially drive the classification: the FI and FF shares are nearly identical whether or not the seven basket tokens are included in the aggregation.


6 Conclusion

This paper proposes a formal, benchmark-independent measure of financial instability for crypto-asset markets. Applying the quantile regression framework of Baur & Schulze (2009) to 137 crypto-assets in 18-month rolling windows, we construct time-varying indices of Financial Instability and Financial Fragility that classify each asset according to whether its sensitivity to a GARCH-standardised systematic shock increases in the left or right tail of its return distribution. Over 2019–2026, approximately 10% of crypto-assets exhibit Financial Instability and 16% exhibit Financial Fragility on average. Financial Instability is episodic, concentrating around the 2022 Crypto-Winter and the SVB episode, while Financial Fragility is more persistent and peaks during the 2021 bull market — a sequencing consistent with the Minsky hypothesis that prolonged expansions accumulate financial fragilities that subsequently unwind. A comparison with 143 individual equity stocks confirms that crypto markets are structurally more unstable and fragile than equity markets, with the exception of the Covid-19 episode. A predictive portfolio exercise further validates the classification: assets satisfying financial stability conditions generate higher cumulative returns over the full sample than assets exhibiting tail amplification in either direction.

Several limitations warrant acknowledgement. The rolling window design introduces a detection lag of approximately half a window length. The systematic shock is constructed from a fixed set of seven benchmark assets; while the robustness analysis confirms that excluding these constituents does not alter the aggregate indices, alternative factor specifications remain to be explored. The minimum-coverage filter may also exclude recently listed tokens during the periods when they are most actively traded.

Several directions for future research follow naturally. First, the rolling Financial Instability index could be integrated into a macroprudential monitoring framework alongside on-chain metrics and stablecoin flows. Second, the asset-level classification opens the door to cross-sectional analyses examining whether Financial Instability is predicted by token-specific characteristics such as market depth, exchange concentration, or smart-contract exposure. Third, the framework could be extended to decentralised finance (DeFi) protocols and stablecoins. Finally, the documented divergence between Financial Instability and Fragility — one episodic and crisis-driven, the other persistent and boom-driven — raises questions about the interaction between the two dimensions over the full boom-bust cycle that a structural model would be well positioned to address.


Appendix

Summary Statistics

(See Table 1 in Section 2.1.)

Crypto-Factor and Systematic Shocks

(See Figures A1 and A2 in Section 2.2.)

Equity Indices

data.frame(
  Ticker = c("^INX","^GSPTSE","^COLCAP","^SPCLXIGPA","^BVSP","^MXX","^SPBLPGPT","^IBG",
             "^GDAX","^FCHI","^FTSE","^FTMIB","^SSMI","^AEX","^OMXS30","^OMXHPI",
             "^OMXCBPI","^DFMGI","^IRTS","^IBEX","^ATX","^BFX","^WIG","^BETI","^PX",
             "^OBXP","^ISEQ","^PSI20","^TRXFLDILP",
             "^JPXNK400","^SSEC","^HSI","^KS11","^BSESN","^NZ50C","^XU100","^AXJO",
             "^STI","^KLSE","^SET100","^JKSE","^VNI","^PSI",
             "^TASI","^JALSH","^EGX30","^NGSEINDEX",".dMIBD00000P"),
  Index = c("S&P 500 (USA)","S&P/TSX Composite (Canada)","MSCI COLCAP (Colombia)",
            "S&P/CLX IGPA (Chile)","Bovespa (Brazil)","IPC (Mexico)",
            "S&P/BVL General (Peru)","S&P/BYMA (Argentina)",
            "DAX (Germany)","CAC 40 (France)","FTSE 100 (UK)","FTSE MIB (Italy)",
            "SMI (Switzerland)","AEX (Netherlands)","OMX Stockholm 30 (Sweden)",
            "OMX Helsinki (Finland)","OMX Copenhagen (Denmark)","Dubai Fin. Market (UAE)",
            "RTS Index (Russia)","IBEX 35 (Spain)","ATX (Austria)","BEL 20 (Belgium)",
            "Warsaw General (Poland)","BET (Romania)","PX Prague (Czech Rep.)",
            "Oslo OBX (Norway)","ISEQ All Share (Ireland)","PSI-20 (Portugal)","FTSE Israel",
            "JPX-Nikkei 400 (Japan)","Shanghai Composite (China)","Hang Seng (Hong Kong)",
            "KOSPI (South Korea)","BSE Sensex (India)","S&P/NZX 50 (New Zealand)",
            "BIST 100 (Turkey)","S&P/ASX 200 (Australia)","Straits Times (Singapore)",
            "FTSE Bursa KLCI (Malaysia)","SET 100 (Thailand)","IDX Composite (Indonesia)",
            "Ho Chi Minh (Vietnam)","PSEi (Philippines)",
            "Tadawul All Share (Saudi Arabia)","FTSE/JSE All Share (South Africa)",
            "EGX 30 (Egypt)","Nigeria All Share","MSCI Bangladesh")
) %>%
  kable(caption = "**Table A1 — Equity Price Indices Used to Construct the Global Equity Factor**",
        col.names = c("Ticker","Index")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 12) %>%
  footnote(general = "The first principal component of these 49 daily price index returns is used as the global equity factor. Russia (^IRTS) is included over the full sample; observations after trading suspension are excluded.")
**Table A1 — Equity Price Indices Used to Construct the Global Equity Factor**
Ticker Index
^INX S&P 500 (USA)
^GSPTSE S&P/TSX Composite (Canada)
^COLCAP MSCI COLCAP (Colombia)
^SPCLXIGPA S&P/CLX IGPA (Chile)
^BVSP Bovespa (Brazil)
^MXX IPC (Mexico)
^SPBLPGPT S&P/BVL General (Peru)
^IBG S&P/BYMA (Argentina)
^GDAX DAX (Germany)
^FCHI CAC 40 (France)
^FTSE FTSE 100 (UK)
^FTMIB FTSE MIB (Italy)
^SSMI SMI (Switzerland)
^AEX AEX (Netherlands)
^OMXS30 OMX Stockholm 30 (Sweden)
^OMXHPI OMX Helsinki (Finland)
^OMXCBPI OMX Copenhagen (Denmark)
^DFMGI Dubai Fin. Market (UAE)
^IRTS RTS Index (Russia)
^IBEX IBEX 35 (Spain)
^ATX ATX (Austria)
^BFX BEL 20 (Belgium)
^WIG Warsaw General (Poland)
^BETI BET (Romania)
^PX PX Prague (Czech Rep.)
^OBXP Oslo OBX (Norway)
^ISEQ ISEQ All Share (Ireland)
^PSI20 PSI-20 (Portugal)
^TRXFLDILP FTSE Israel
^JPXNK400 JPX-Nikkei 400 (Japan)
^SSEC Shanghai Composite (China)
^HSI Hang Seng (Hong Kong)
^KS11 KOSPI (South Korea)
^BSESN BSE Sensex (India)
^NZ50C S&P/NZX 50 (New Zealand)
^XU100 BIST 100 (Turkey)
^AXJO S&P/ASX 200 (Australia)
^STI Straits Times (Singapore)
^KLSE FTSE Bursa KLCI (Malaysia)
^SET100 SET 100 (Thailand)
^JKSE IDX Composite (Indonesia)
^VNI Ho Chi Minh (Vietnam)
^PSI PSEi (Philippines)
^TASI Tadawul All Share (Saudi Arabia)
^JALSH FTSE/JSE All Share (South Africa)
^EGX30 EGX 30 (Egypt)
^NGSEINDEX Nigeria All Share
.dMIBD00000P MSCI Bangladesh
Note:
The first principal component of these 49 daily price index returns is used as the global equity factor. Russia (^IRTS) is included over the full sample; observations after trading suspension are excluded.

Equity Stocks

stocks_tbl <- data.frame(
  Region   = c(rep("North America",1), rep("Europe",9), rep("Asia-Pacific",5), "Total"),
  Country  = c("United States","United Kingdom","Germany","France","Netherlands",
               "Italy","Spain","Switzerland","Sweden","Denmark",
               "Japan","Hong Kong","China (US-listed)","Australia","India",""),
  Exchange = c("NYSE/NASDAQ","LSE","XETRA","Euronext Paris","Euronext Amsterdam",
               "Borsa Italiana","BME","SIX","Nasdaq Stockholm","Nasdaq Copenhagen",
               "TSE","HKEX","NYSE/NASDAQ","ASX","NSE",""),
  N        = c(60,10,10,10,5,5,4,4,4,2,10,5,3,7,4,143)
)

stocks_tbl %>%
  kable(caption = "**Table A2 — Individual Equity Stocks: Composition by Country**",
        col.names = c("Region","Country","Exchange","N"),
        align = c("l","l","l","r")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 12) %>%
  row_spec(nrow(stocks_tbl), bold = TRUE) %>%
  collapse_rows(columns = 1, valign = "middle")
**Table A2 — Individual Equity Stocks: Composition by Country**
Region Country Exchange N
North America United States NYSE/NASDAQ 60
Europe United Kingdom LSE 10
Germany XETRA 10
France Euronext Paris 10
Netherlands Euronext Amsterdam 5
Italy Borsa Italiana 5
Spain BME 4
Switzerland SIX 4
Sweden Nasdaq Stockholm 4
Denmark Nasdaq Copenhagen 2
Asia-Pacific Japan TSE 10
Hong Kong HKEX 5
China (US-listed) NYSE/NASDAQ 3
Australia ASX 7
India NSE 4
Total 143

Session Information

sessionInfo()
R version 4.5.0 (2025-04-11 ucrt)
Platform: x86_64-w64-mingw32/x64
Running under: Windows 11 x64 (build 22631)

Matrix products: default
  LAPACK version 3.12.1

locale:
[1] LC_COLLATE=French_France.utf8  LC_CTYPE=French_France.utf8    LC_MONETARY=French_France.utf8
[4] LC_NUMERIC=C                   LC_TIME=French_France.utf8    

time zone: Europe/Paris
tzcode source: internal

attached base packages:
[1] stats     graphics  grDevices utils     datasets  methods   base     

other attached packages:
 [1] zoo_1.8-14        progress_1.2.3    data.table_1.17.4 quantreg_6.1      SparseM_1.84-2    purrr_1.1.0      
 [7] yaml_2.3.10       lubridate_1.9.4   scales_1.4.0      kableExtra_1.4.0  ggplot2_4.0.0     tidyr_1.3.1      
[13] dplyr_1.1.4       arrow_21.0.0.1   

loaded via a namespace (and not attached):
 [1] gtable_0.3.6       xfun_0.52          bslib_0.9.0        lattice_0.22-6     tzdb_0.5.0        
 [6] vctrs_0.6.5        tools_4.5.0        generics_0.1.4     tibble_3.2.1       pkgconfig_2.0.3   
[11] Matrix_1.7-3       RColorBrewer_1.1-3 S7_0.2.0           assertthat_0.2.1   lifecycle_1.0.4   
[16] compiler_4.5.0     farver_2.1.2       stringr_1.5.1      MatrixModels_0.5-4 textshaping_1.0.1 
[21] htmltools_0.5.8.1  sass_0.4.10        pillar_1.10.2      crayon_1.5.3       jquerylib_0.1.4   
[26] MASS_7.3-65        rsconnect_1.5.1    cachem_1.1.0       tidyselect_1.2.1   digest_0.6.37     
[31] stringi_1.8.7      labeling_0.4.3     splines_4.5.0      fastmap_1.2.0      grid_4.5.0        
[36] cli_3.6.5          magrittr_2.0.3     survival_3.8-3     withr_3.0.2        prettyunits_1.2.0 
[41] bit64_4.6.0-1      timechange_0.3.0   rmarkdown_2.29     bit_4.6.0          hms_1.1.3         
[46] evaluate_1.0.3     knitr_1.50         viridisLite_0.4.2  rlang_1.1.6        glue_1.8.0        
[51] xml2_1.3.8         svglite_2.2.2      rstudioapi_0.17.1  jsonlite_2.0.0     R6_2.6.1          
[56] systemfonts_1.3.1 

Florian Kraus — University of Bordeaux (BSE) — Compiled with R 4.5.0 and knitr 1.50.


  1. This filter removes extreme observations likely caused by thin trading or data errors, affecting less than 0.5% of the final sample.↩︎

  2. We apply an eGARCH(1,1) model with Student distribution, selected as the optimal fit by AIC and BIC. We use GARCH rather than ARMA because residuals from an AR fit exhibit significant ARCH effects.↩︎

  3. Alternative scalings we tested: (i) using raw GARCH residuals εt as regressor (between-window distributional inequality); (ii) rolling-window variance scaling of both the shock and returns (still contaminated by slow volatility cycles); (iii) filtering both returns and shock by GARCH (removes the variance channel from the dependent variable). The GARCH-standardized shock zt = εt/σt is the preferred specification.↩︎

  4. A typical 18-month window contains approximately 547 trading days, of which around 27 are extreme negative and 27 are extreme positive shock days.↩︎

---
title: |
  **Who Stays Calm in Chaos?**
  The Financial Instability of Crypto-Assets
author: "Florian Kraus — University of Bordeaux (BSE)"
date: "January 2026"
output:
  html_notebook:
    theme: flatly
    highlight: tango
    toc: true
    toc_float: false
    toc_depth: 3
    number_sections: true
    code_folding: show
    mathjax: null
    includes:
      in_header: mathjax_header.html
    css: notebook_style.css
---

```{r setup, include=FALSE}
knitr::opts_chunk$set(
  echo      = TRUE,
  message   = FALSE,
  warning   = FALSE,
  fig.align = "center",
  fig.width = 12,
  fig.height = 6,
  out.width = "100%"
)
```

```{r init}
# ── Packages ───────────────────────────────────────────────────────────────────
suppressPackageStartupMessages({
  library(arrow)
  library(dplyr)
  library(tidyr)
  library(ggplot2)
  library(kableExtra)
  library(scales)
  library(lubridate)
  library(yaml)
})

# ── Project helpers and paths ──────────────────────────────────────────────────
source("src/utils/helpers.R")
paths <- load_paths()

# Convenience: garch_shock_ext specification paths
sp <- paths$specs$garch_shock_ext
```

---

> *This notebook reproduces the full empirical analysis of "Who Stays Calm in Chaos? The Financial Instability of Crypto-Assets." Each section integrates the paper text with the R code that produced it. Heavy estimations (DFM, GARCH, rolling QR) are wrapped in `eval=FALSE` blocks; the notebook loads results from pre-computed parquet files in `output/models/`.*

---

# Abstract {.unnumbered}

This article measures financial instability in crypto-asset markets using the quantile regression framework of Baur & Schulze (2009), extended to individual assets and estimated in 18-month rolling windows. We classify each asset as financially unstable or fragile based on whether its sensitivity to systematic shocks increases in the left or right tail of its return distribution. Applied to 137 crypto-assets over 2019–2026, approximately **10%** of assets exhibit Financial Instability and **16%** exhibit Financial Fragility on average. Financial Instability concentrates around the 2022 Crypto-Winter while Financial Fragility peaks during the 2021 bull market, consistent with the Minsky hypothesis. Crypto markets exhibit substantially higher instability and fragility than a benchmark of individual equity stocks. A predictive portfolio exercise confirms that stable assets generate higher cumulative returns than tail-amplifying assets over the full sample.

**Keywords:** Financial Stability · Crypto-Assets · Contagion · Systematic Risk · Quantile

---

# Introduction

Crypto-assets have expanded rapidly in both popularity and market capitalization, leading to their gradual inclusion in investment portfolios. Yet, this growing adoption raises concerns about growing interactions with traditional markets, and the financial stability of crypto-markets is a major concern. Existing works fail to produce objective conclusions on the financial stability of crypto-markets. This paper addresses this gap by adopting an operational definition of financial asset stability from Baur & Schulze (2009) that yields a binary classification for each crypto-asset in each period.

While early adoption of crypto-assets remained limited to a niche group of enthusiasts, large holders ("whales"), and individuals engaged in illegal activities (Athey et al., 2016; Foley et al., 2019; Aiello et al., 2023), the creation of crypto-exchanges accelerated the development of crypto-markets since 2017. Several works document the dynamics of retail adoption (Lammer et al., 2020; Auer et al., 2022; Cornelli et al., 2023). On the institutional side, research consistently shows rising adoption (Huang et al., 2022; Auer & Ongena, 2022). This growing adoption raises concerns from regulators and academics regarding bubble dynamics (Panetta, 2022; Kyriazis et al., 2020) and the financial stability of crypto-markets more broadly (Grippa & Mann, 2022). However, existing measures do not provide a financial stability index grounded in a formal definition and independent of benchmark selection.

Financial market instability received greater attention in the aftermath of the Bretton-Woods system collapse (Minsky, 1977). The literature has converged on several complementary notions of financial stability. From a practical perspective, it represents the absence of financial instabilities or crises (Crockett, 1996; Chant, 2003). The Financial Instability Hypothesis (Minsky, 1977, 1992) argues that financial instability is endogenous: prolonged favorable conditions foster increasingly fragile financial structures. Schinasi (2004) defines financial stability as the capacity of a financial system to fulfil its functions resiliently. Allen & Gale (2008) argue that financial instabilities transform markets from shock absorbers to shock amplifiers. Against this backdrop, Baur & Schulze (2009) propose testing financial stability via quantile regressions.

The literature examining the financial stability of crypto-assets is extensive, yet only a few studies follow established definitions. Some articles use volatility comparisons (Bhosale & Mavale, 2018; Gupta et al., 2022; Baur & Dimpfl, 2021; Kristoufek, 2023), which remain sensitive to benchmark choice and are not a direct measure of financial stability. More recent works (Chatziantoniou et al., 2021; Ando et al., 2022; Bouri et al., 2021) introduce tail-sensitive connectedness measures, yet often rely on ad hoc criteria to label stability without enabling formal coefficient comparisons across quantiles.

This article proposes to fill this gap by employing the quantile regression framework of Baur & Schulze (2009) at the individual asset level for crypto-markets. Our **three contributions** are:

1. **Financial Instability as left-tail amplification.** We propose a definition-grounded metric — the amplification of negative systematic shocks — rather than benchmark-dependent volatility measures. Approximately 10% of crypto-assets amplify systematic shocks in their extreme negative regimes on average.

2. **Asset-level, rolling-window application.** Financial Instability is episodic, concentrating in three structurally distinct waves: the 2017–2018 Crypto-Winter, the Terra/LUNA–FTX sequence of 2022, and a third episode in late 2025.

3. **Financial Fragility index.** Following the Financial Instability Hypothesis, we construct an index of upside amplification as an indicator of speculative build-up. Financial Fragility peaks during the 2021 bull market, consistent with Minsky (1977, 1992).

The rest of the article is organized as follows. Section 2 presents the data and methodology. Section 3 provides our results. Section 4 reports further analyses. Section 5 presents robustness checks. Section 6 concludes.

---

# Methodology and Data {#methodology}

## Data {#data}

We collect the list of the 40 most capitalized crypto-assets on the first week of each year from 2017 to 2025 from CoinMarketCap and remove stablecoins from the sample, leading to a final sample of **137 crypto-assets**. Stablecoins are identified using CoinGecko and excluded under categories "Stablecoins," "Fiat-backed Stablecoins," "USD Stablecoins," and "Crypto-backed Stablecoins."

We extract daily price data from **January 1, 2018 to February 1, 2026** from CryptoCompare. Returns are measured as log-price differences. To ensure liquidity, we remove observations when the volume is below 10 tokens.^[This filter removes extreme observations likely caused by thin trading or data errors, affecting less than 0.5% of the final sample.]

```{r load-data}
# ── Load the final crypto panel ────────────────────────────────────────────────
# data_crypto.parquet contains log returns, market caps, and — crucially —
# resid_garch_std: the GARCH-standardised systematic shock z_t = ε_t/σ_t.
data_crypto <- arrow::read_parquet(paths$data$final$data_crypto) %>%
  mutate(date = as.Date(date)) %>%
  select(date, variable, return, resid_garch_std)

cat("Panel:", nrow(data_crypto), "obs —",
    length(unique(data_crypto$variable)), "tokens —",
    format(min(data_crypto$date)), "to", format(max(data_crypto$date)), "\n")
```

```{r summary-stats-table}
# ── Summary statistics ─────────────────────────────────────────────────────────
make_stats_row <- function(x, label, n_assets = NA_character_) {
  data.frame(
    Series       = label,
    Mean         = round(mean(x, na.rm=TRUE), 3),
    `Std. Dev.`  = round(sd(x, na.rm=TRUE), 3),
    Median       = round(median(x, na.rm=TRUE), 3),
    Minimum      = round(min(x, na.rm=TRUE), 3),
    Maximum      = round(max(x, na.rm=TRUE), 3),
    Observations = formatC(sum(!is.na(x)), format="d", big.mark=","),
    Assets       = n_assets,
    check.names  = FALSE
  )
}

ret_vec   <- data_crypto$return[!is.na(data_crypto$return)]
shock_vec <- data_crypto %>%
  filter(!is.na(resid_garch_std)) %>%
  distinct(date, resid_garch_std) %>%
  pull(resid_garch_std)

bind_rows(
  make_stats_row(ret_vec,   "Log returns",      n_assets = as.character(n_distinct(data_crypto$variable))),
  make_stats_row(shock_vec, "Systematic shock",  n_assets = "—")
) %>%
  kable(caption = "**Table 1 — Summary Statistics**",
        align = c("l","r","r","r","r","r","r","r")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 13) %>%
  footnote(
    general = "Log returns are daily log price changes for 137 crypto-assets over January 2018 to February 2026. The systematic shock is the standardised residual z_t = ε_t/σ_t of an eGARCH(1,1) model applied to the first-differenced Crypto-Factor.",
    general_title = "Note: "
  )
```

## Systematic Shocks {#shocks}

To estimate systematic shocks in crypto-markets, we specify a Dynamic Factor Model (DFM) for a panel of crypto-asset prices. As Rey & Miranda-Agrippino (2021) argue, the single factor of equity prices combines market risk and aggregate risk-aversion. We maintain this interpretation for crypto markets.

Let $p_t$ denote an $n$-dimensional vector of daily prices $p_{i,t}$ that are a linear combination of a common factor $f_t$ following an $AR(p)$ process and an idiosyncratic error term $\varepsilon_{i,t}$:

$$p_{i,t} = \lambda_i(L)\, f_t + \varepsilon_{i,t}$$

$$f_t = A_1 f_{t-1} + \cdots + A_q f_{t-q} + \eta_t$$

where $\eta_t \sim \mathcal{N}(0, \Sigma)$ and $\lambda_i$ is a vector of factor loadings for asset $i$. Following Che et al. (2023), we estimate the single factor on the **7 biggest crypto-assets** by market capitalization (Bitcoin, Ethereum, Ripple, Binance Coin, Cardano, TRON and DOGE), representing around 70% of the crypto-market capitalization at the end of our sample.

```{r estimate-dfm, eval=FALSE}
# ── DFM estimation (eval=FALSE — run once via src/02_factor_and_shocks.R) ─────
# Inputs:  data/raw/* price data from CryptoCompare
# Outputs: data/intermediate/market_shocks.parquet  (contains the DFM factor
#          series, raw GARCH residuals, and resid_garch_std_crypto = z_t)
#
# The eGARCH(1,1) with Student distribution is then applied in
# src/02c_garch_std_shock.R, which patches data_crypto.parquet to add
# resid_garch_std = z_t = ε_t / σ_t.
source("src/02_factor_and_shocks.R")
source("src/02c_garch_std_shock.R")
```

We construct the systematic shock $f^*_t$ as the **standardised residuals** of an **eGARCH(1,1) model**^[We apply an eGARCH(1,1) model with Student distribution, selected as the optimal fit by AIC and BIC. We use GARCH rather than ARMA because residuals from an AR fit exhibit significant ARCH effects.] applied to the Crypto-Factor in first difference for stationarity:

$$f^*_t = z_t = \frac{\varepsilon_t}{\sigma_t}$$

where $\varepsilon_t$ is the GARCH residual and $\sigma_t$ is the estimated conditional standard deviation. By construction, $z_t$ has unit scale at every date, making cross-window shock distributions comparable. This matters because the GARCH residuals exhibit substantial variation in conditional variance across subsamples: a unit shock in a calm window is not comparable to a unit shock in a turbulent window. Standardising by $\sigma_t$ removes this scale heterogeneity while preserving the variance-transmission channel in the dependent variable — a key distinction from alternative scalings that filter the returns themselves.^[Alternative scalings we tested: (i) using raw GARCH residuals $\varepsilon_t$ as regressor (between-window distributional inequality); (ii) rolling-window variance scaling of both the shock and returns (still contaminated by slow volatility cycles); (iii) filtering both returns and shock by GARCH (removes the variance channel from the dependent variable). The GARCH-standardized shock $z_t = \varepsilon_t/\sigma_t$ is the preferred specification.]

```{r load-and-plot-shocks, fig.cap="**Figure A1 (Appendix) — Systematic shocks in crypto-markets.** Daily innovations to the crypto market factor: standardised residuals $z_t = \\varepsilon_t/\\sigma_t$ of an eGARCH(1,1) model fitted to the Crypto-Factor series in first-difference. These shocks constitute the explanatory variable in all quantile regressions.", fig.height=4}
# z_t is stored directly in data_crypto.parquet (column resid_garch_std)
# It is also in data/intermediate/market_shocks.parquet as resid_garch_std_crypto

shock_series <- data_crypto %>%
  distinct(date, z_t = resid_garch_std) %>%
  filter(!is.na(z_t)) %>%
  arrange(date)

ggplot(shock_series, aes(x = date, y = z_t)) +
  geom_line(color = "black", linewidth = 0.4, alpha = 0.85) +
  geom_hline(yintercept = 0, linetype = "dashed", color = "grey50") +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.01)) +
  labs(x = NULL, y = expression(z[t] == epsilon[t] / sigma[t])) +
  theme_crypto()
```

```{r plot-factor, fig.cap="**Figure A2 (Appendix) — Crypto-Factor (2018–2026).** Estimated as the first principal component of 7 crypto-asset prices (BTC, ETH, XRP, BNB, ADA, TRX, DOGE) with a Dynamic Factor Model.", fig.height=4}
# The factor series is saved in market_shocks.parquet
market_shocks <- arrow::read_parquet(paths$data$intermediate$market_shocks) %>%
  mutate(date = as.Date(date))

ggplot(market_shocks, aes(x = date, y = serie)) +
  geom_line(color = "black", linewidth = 0.6) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.01)) +
  labs(x = NULL, y = "Factor level") +
  theme_crypto()
```

## Quantile Regression Framework {#quantilereg}

Our empirical strategy builds on Baur & Schulze (2009), who define financial asset stability as the absence of increasing shock sensitivity in the tails of the return distribution relative to normal conditions. For each asset $i$, we estimate the following quantile regression at three quantiles $\tau$ — the lower tail ($\tau = 0.05$), the median ($\tau = 0.50$), and the upper tail ($\tau = 0.95$):

$$Q_\tau(r_{it} \mid f^*_t) = \alpha_i(\tau) + \beta_i(\tau)\, f^*_t + \gamma_i(\tau)\, f^*_t \cdot D_t^{\text{lower}} + \phi_i(\tau)\, f^*_t \cdot D_t^{\text{upper}} + \varepsilon_i(\tau)$$

where $r_{it}$ is the return of asset $i$, $f^*_t = z_t$ is the standardised systematic shock, and $D_t^{\text{lower}}$ ($D_t^{\text{upper}}$) is a dummy equal to one when $f^*_t$ falls in the bottom (top) 5% of its in-window distribution.

The bilateral shock dummies isolate the relationship between **asset return regimes** and systematic shocks of ordinary magnitude. Without them, $\beta_i(\tau)$ would conflate the asset's intrinsic regime-dependent sensitivity with the general co-movement induced by tail market events — confounding tail-dependence with co-exceedances. Under this setup, $\beta_i(\tau)$ identifies how asset return regimes relate to the systematic factor under **normal market conditions**, while $\gamma_i$ and $\phi_i$ capture additional sensitivity under extreme shocks.

The equation delivers three quantile-specific estimates: $\hat{\beta}^{\text{lower}} \equiv \hat{\beta}_i(0.05)$, $\hat{\beta}^{\text{med}} \equiv \hat{\beta}_i(0.50)$, and $\hat{\beta}^{\text{upper}} \equiv \hat{\beta}_i(0.95)$. Each tail is tested against the median via a **pairwise Wald test** (Koenker & Machado, 1999) at the **10% significance level**.

**Classification rules:**

- **Financial Instability** (left-tail amplification): $\hat{\beta}^{\text{lower}} > \hat{\beta}^{\text{med}}$ and Wald test rejects at 10%. Shock transmission intensifies when assets are in distress, amplifying aggregate downside risk (Forbes & Rigobon, 2001; Longin & Solnik, 2001).

- **Financial Fragility** (right-tail amplification): $\hat{\beta}^{\text{upper}} > \hat{\beta}^{\text{med}}$ and Wald test rejects at 10%. Assets amplifying positive market shocks during expansionary phases fuel price bubbles and create correlated exposures that subsequently unwind (Brunnermeier et al., 2013), consistent with the Minsky hypothesis.

To capture time variation, we estimate the equation within **18-month rolling windows**, stepping monthly. Assets are included only if their return coverage exceeds 90% of the available trading days in a given window.^[A typical 18-month window contains approximately 547 trading days, of which around 27 are extreme negative and 27 are extreme positive shock days.] Each window is self-contained: all estimates and shock quantile thresholds are computed within the window, preventing contamination by out-of-window information.

```{r rolling-qr, eval=FALSE}
# ── Rolling QR estimation (eval=FALSE — run once via src/25_garch_shock_full.R)
#
# The function run_rolling_qr_analysis() in helpers.R parallelises over windows.
# It reads data_crypto.parquet (which already contains resid_garch_std = z_t),
# and applies the formula:
#   return ~ resid_garch_std + resid_garch_std:d_low + resid_garch_std:d_high
# with bilateral dummies (add_dummies = "bilateral", shock_q = 0.05).
#
# Output: output/models/roll_18_garch_shock_ext.parquet
#   Columns: end, variable, left_tail, right_tail,
#            beta_lower, beta_median, beta_upper,
#            p_lower, p_upper, instable, cont, spec
source("src/25_garch_shock_full.R")
```

```{r load-rolling}
# ── Load pre-computed rolling QR results ──────────────────────────────────────
roll <- arrow::read_parquet(sp$models$roll_18) %>%
  mutate(end = as.Date(end))

cat("Rolling results:", nrow(roll), "rows —",
    length(unique(roll$variable)), "tokens —",
    length(unique(roll$end)), "windows\n")
cat("Classification columns: cont (FI), spec (FF), instable\n")
```

```{r n-assets-plot, fig.cap="**Figure A3 (Appendix) — Number of assets in the rolling estimation sample.** An asset is included if its return coverage within the window is at least 90%. The figure illustrates the expansion of the crypto universe over the sample period.", fig.height=4}
roll %>%
  group_by(end) %>%
  summarise(n = n_distinct(variable), .groups = "drop") %>%
  ggplot(aes(x = end, y = n)) +
  geom_line(color = "black", linewidth = 0.7) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.01)) +
  scale_y_continuous(limits = c(0, NA), expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Number of assets") +
  theme_crypto()
```

---

# Results {#results}

## Financial Instability and Financial Fragility Dynamics

Figure 1 reports the results of our classification strategy in rolling windows, alongside crypto market capitalisation. On average, approximately **10% of crypto-assets** exhibit left-tail amplification of systematic shocks. The fraction exhibiting **Financial Fragility** is higher, averaging 15%.

```{r compute-shares}
# ── Rolling shares ─────────────────────────────────────────────────────────────
shares <- roll %>%
  group_by(end) %>%
  summarise(
    fi   = mean(cont, na.rm = TRUE) * 100,
    ff   = mean(spec, na.rm = TRUE) * 100,
    n    = n_distinct(variable),
    .groups = "drop"
  )

shares %>%
  summarise(
    `FI mean (%)`  = round(mean(fi), 1),
    `FI max (%)`   = round(max(fi), 1),
    `FF mean (%)`  = round(mean(ff), 1),
    `FF max (%)`   = round(max(ff), 1),
    `Avg. N`       = round(mean(n))
  ) %>%
  kable(caption = "**Average FI/FF shares across 80 rolling windows**") %>%
  kable_styling(bootstrap_options = c("striped","hover"), full_width = FALSE)
```

```{r fig-baseline, fig.cap="**Figure 1 — Financial Instability and Financial Fragility dynamics.** Fraction of crypto-assets exhibiting increasing left-tail dependence (Financial Instability) and right-tail dependence (Financial Fragility) to the systematic shock. Rolling 18-month windows, monthly step.", fig.height=6}
shares %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(
    recode(metric, fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")
  )) %>%
  ggplot(aes(x = end, y = share / 100, color = metric, linetype = metric)) +
  geom_line(linewidth = 1) +
  scale_color_manual(
    values = c("Financial Instability" = "#d62728",
               "Financial Fragility"   = "#1f77b4"),
    name = NULL) +
  scale_linetype_manual(
    values = c("Financial Instability" = "solid",
               "Financial Fragility"   = "solid"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

**The Financial Instability share is episodic rather than persistent.** Three structurally distinct episodes emerge:

1. **2019**: tail of the 2017–2018 Crypto-Winter, following Bitcoin's peak near \$20,000 in December 2017 and a subsequent decline exceeding 80%.

2. **Mid-2022** (peak: November 2022): the Terra/LUNA implosion in May 2022, followed by the failures of Three Arrows Capital and Celsius Network, and the FTX exchange collapse in November 2022. A secondary elevation persists into early 2023, coinciding with the Silicon Valley Bank failure and the brief USDC depeg.

3. **Late 2025**: a third episode at the end of the estimation sample.

**The Financial Fragility share exhibits a different dynamic**: elevated and relatively persistent over 2020–2024. It peaks above 30% in mid-2021 during the bull market following the initial Covid-19 recovery, consistent with Lahmiri & Bekiros (2020), and remains above its sample mean through 2024. This pattern is consistent with Minsky's Financial Instability Hypothesis (1992): speculative financing and herding accumulate during sustained price appreciation, manifesting as right-tail amplification. The increase in Financial Fragility slightly precedes the market capitalisation peak, consistent with behavioural amplification preceding aggregate overvaluation (Brunnermeier & Nagel, 2009; Stolborg et al., 2025).

Neither index co-moves systematically with aggregate market volatility over the full sample. For Financial Instability, the absence of a positive correlation with volatility is consistent with studies arguing that volatility is not a good predictor of financial crises (Danielsson et al., 2018; Sornette & Cauwels, 2018) and justifies our approach.

**Financial Fragility exceeds Financial Instability throughout the sample**, reflecting the asymmetric structure of crypto market cycles. Boom phases are prolonged and broad-based relative to sharp but brief crash episodes: during expansionary phases, speculative demand and momentum trading cause a large cross-section of assets to amplify positive systematic shocks simultaneously (Minsky, 1977; Bouri et al., 2019). Crash episodes, by contrast, are concentrated in time and the amplification of negative shocks is more selective. The result is a persistently higher share of assets exhibiting right-tail amplification than left-tail amplification.

---

# Further Analysis {#further}

## Portfolio Performance {#portfolio}

We construct four equally-weighted portfolios formed on the rolling classification, rebalanced monthly with a **one-month forward lag** to ensure the classification is fully out-of-sample:

- **Financial Instability** portfolio: holds assets with $\hat{\beta}_i(0.05) > \hat{\beta}_i(0.50)$ exclusively
- **Financial Fragility** portfolio: holds assets with $\hat{\beta}_i(0.95) > \hat{\beta}_i(0.50)$ exclusively
- **Stable** portfolio: all remaining assets satisfying neither criterion
- **Top-10** benchmark: equally-weighted portfolio of the ten most prominent crypto-assets

```{r portfolio-code, eval=FALSE}
# ── Portfolio construction (eval=FALSE — run once via src/12_portfolio_performance.R)
# Uses roll_18_garch_shock_ext.parquet + daily return data from data_crypto.parquet.
# Classification at window end t → portfolio held over month t+1.
# Output: output/figures/garch_shock_ext/paper/portfolio_predictive.pdf
source("src/12_portfolio_performance.R")
```

```{r fig-portfolio, fig.cap="**Figure 2 — Portfolio Performance across Categories.** Cumulative wealth (base 100) on a logarithmic scale. The *Financial Instability* portfolio holds pure left-tail amplifiers; the *Financial Fragility* portfolio holds pure right-tail amplifiers; the *Stable* portfolio holds all remaining assets. The *Top-10* benchmark is an equally-weighted basket of the ten most prominent crypto-assets. Portfolios are rebalanced monthly with a one-month forward lag (out-of-sample).", fig.height=6}

# ── Helper functions (from src/12_portfolio_performance.R) ────────────────────
build_constituents <- function(roll_df, portfolio_type) {
  if (portfolio_type == "bench") {
    return(roll_df %>% distinct(end, variable) %>% mutate(in_portfolio = TRUE))
  }
  roll_df %>%
    distinct(end, variable, instable, cont, spec) %>%
    mutate(in_portfolio = case_when(
      portfolio_type == "cont_only" ~ (as.logical(cont) & !as.logical(spec)),
      portfolio_type == "spec_only" ~ (as.logical(spec) & !as.logical(cont)),
      portfolio_type == "stable"    ~ (!as.logical(instable) & !is.na(instable)),
      TRUE ~ FALSE
    )) %>%
    select(end, variable, in_portfolio)
}

build_constituents_lagged <- function(roll_df, portfolio_type) {
  build_constituents(roll_df, portfolio_type) %>%
    arrange(variable, end) %>%
    group_by(variable) %>%
    mutate(in_portfolio = lag(in_portfolio, default = FALSE)) %>%
    ungroup()
}

compute_daily_returns <- function(daily_df, constituents_df) {
  dates_sched <- sort(unique(constituents_df$end))
  ret_cap     <- log(2)
  purrr::map_dfr(seq_along(dates_sched), function(i) {
    t_start <- dates_sched[i]
    t_end   <- if (i < length(dates_sched)) dates_sched[i + 1L] else max(daily_df$date)
    members <- constituents_df %>% filter(end == t_start, in_portfolio) %>% pull(variable)
    rows    <- daily_df %>% filter(date > t_start, date <= t_end)
    if (length(members) == 0)
      return(rows %>% distinct(date) %>% mutate(port_ret = NA_real_, n_assets = 0L))
    wts <- tibble(variable = members, weight = 1 / length(members))
    rows %>%
      filter(variable %in% members) %>%
      left_join(wts, by = "variable") %>%
      mutate(simple_ret = exp(pmax(pmin(return, ret_cap), -ret_cap)) - 1) %>%
      group_by(date) %>%
      summarise(port_ret  = sum(simple_ret * weight, na.rm = TRUE) /
                              max(sum(weight[!is.na(simple_ret)]), 1e-9),
                n_assets  = sum(!is.na(simple_ret)),
                .groups = "drop")
  })
}

# ── Top-10 fixed basket ────────────────────────────────────────────────────────
TOP10 <- c("BTC","ETH","BNB","XRP","ADA","SOL","DOGE","TRX","LTC","DOT")
top10_assets <- intersect(TOP10, unique(roll$variable))
top10_const  <- roll %>% distinct(end) %>%
  crossing(variable = top10_assets) %>%
  mutate(in_portfolio = TRUE)

# ── Build lagged predictive portfolios ────────────────────────────────────────
port_types <- c(cont_only = "Financial Instability",
                spec_only = "Financial Fragility",
                stable    = "Stable")

port_data <- purrr::imap_dfr(port_types, function(label, ptype) {
  build_constituents_lagged(roll, ptype) %>%
    compute_daily_returns(data_crypto, .) %>%
    mutate(portfolio = label)
})

top10_data <- compute_daily_returns(data_crypto, top10_const) %>%
  mutate(portfolio = "Top-10")

# ── Cumulative wealth ──────────────────────────────────────────────────────────
PORT_LEVELS <- c("Stable", "Top-10", "Financial Fragility", "Financial Instability")
COLORS_PORT <- c("Financial Instability" = "#d62728",
                 "Financial Fragility"   = "#1f77b4",
                 "Stable"                = "darkgreen",
                 "Top-10"                = "black")

bind_rows(port_data, top10_data) %>%
  mutate(portfolio = factor(portfolio, levels = PORT_LEVELS)) %>%
  group_by(portfolio) %>%
  arrange(date) %>%
  mutate(cum_wealth = 100 * cumprod(ifelse(is.na(port_ret), 1, 1 + port_ret))) %>%
  ungroup() %>%
  ggplot(aes(x = date, y = cum_wealth, color = portfolio, linetype = portfolio)) +
  geom_line(linewidth = 0.9, na.rm = TRUE) +
  scale_y_log10(labels = scales::label_number(accuracy = 1)) +
  scale_color_manual(values = COLORS_PORT, name = NULL) +
  scale_linetype_manual(
    values = c("Financial Instability" = "solid", "Financial Fragility" = "solid",
               "Stable" = "solid", "Top-10" = "solid"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  labs(x = NULL, y = "Cumulative wealth (log scale, base 100)") +
  theme_crypto()
```

The **Stable portfolio generates the highest cumulative return** over the full sample, outperforming both the Top-10 benchmark and the tail-amplification portfolios. Assets that do not amplify systematic shocks in either tail are less exposed to sharp drawdowns during stress episodes while still participating in broad market appreciation during calm phases, consistent with Ang et al. (2006). The result is consistent with the long-term performance advantage of assets avoiding crash amplification or bubble formation (Pellegrini et al., 2008).

The Financial Instability and Financial Fragility portfolios **both underperform** over the full sample. Both portfolios expand sharply during the 2021 bull market but decouple from the benchmark during the 2022 Crypto-Winter. This reflects the twin costs of tail amplification: assets exhibiting left-tail sensitivity underperform during downturns through shock amplification, while assets with right-tail sensitivity may overperform short-term during speculative phases but experience reversals once the speculative phase ends.

Overall, our tail-dependence portfolio analysis demonstrates the importance of incorporating tail-dependence in portfolio strategies, consistent with Bergmann et al. (2018) who show that tail dependence models outperform the Markowitz model in terms of cumulative return.

## Comparisons with Equity Markets {#equity}

To benchmark crypto instability against a traditional asset class, we apply the identical rolling quantile regression framework to individual equity stocks. Adjusted daily closing prices are sourced from Yahoo Finance for a universe of **143 large-cap firms** spanning North America (60 stocks), Europe (54 stocks), Japan (13 stocks), China, Australia, and India (4 stocks).

The equity sample begins in January 1995 and closes in February 2026. The systematic equity shock is constructed by applying the same two-step procedure: a first principal component is extracted from **49 country-level equity price indices** sourced from Datastream following Rey & Miranda-Agrippino (2021), and a GARCH(1,1) model is estimated on the first-differenced factor. Individual stocks are not used in factor construction, so the systematic shock is exogenous by design.

```{r equity-code, eval=FALSE}
# ── Equity rolling QR (eval=FALSE — run once via src/25_garch_shock_full.R)
# Uses data/intermediate/stocks_panel.parquet.
# Output: output/models/stocks_roll_garch_shock_ext.parquet
#         output/figures/garch_shock_ext/equity/stocks_comparison.pdf
```

```{r load-equity}
equity <- arrow::read_parquet(sp$models$stocks_roll) %>%
  mutate(end = as.Date(end))

cat("Equity results:", nrow(equity), "rows —",
    length(unique(equity$variable)), "stocks\n")

bind_rows(
  data.frame(Group = "Crypto",
             `FI mean (%)` = round(mean(shares$fi), 1),
             `FF mean (%)` = round(mean(shares$ff), 1)),
  equity %>%
    group_by(end) %>%
    summarise(fi = mean(cont, na.rm=TRUE)*100,
              ff = mean(spec, na.rm=TRUE)*100, .groups="drop") %>%
    summarise(Group = "Equity",
              `FI mean (%)` = round(mean(fi), 1),
              `FF mean (%)` = round(mean(ff), 1))
) %>%
  kable(caption = "**Average Financial Instability and Fragility: Crypto vs. Equity**") %>%
  kable_styling(bootstrap_options = c("striped","hover"), full_width = FALSE)
```

```{r fig-equity, fig.cap="**Figure 3 — Financial Instability and Fragility in Equity Markets.** Rolling FI and FF shares for crypto-assets and individual equity stocks over the overlapping estimation period (July 2019 – January 2026). Each share is the fraction of assets classified as exhibiting left-tail (FI) or right-tail (FF) amplification at the 10% significance level.", fig.height=6}
equity_shares <- equity %>%
  group_by(end) %>%
  summarise(fi = mean(cont, na.rm=TRUE)*100,
            ff = mean(spec, na.rm=TRUE)*100, .groups="drop")

bind_rows(
  shares      %>% select(end, fi, ff) %>% mutate(group = "Crypto"),
  equity_shares                        %>% mutate(group = "Equity")
) %>%
  filter(end >= as.Date("2019-07-01")) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
                           fi = "Financial Instability",
                           ff = "Financial Fragility"),
                    levels = c("Financial Instability", "Financial Fragility")),
    group  = factor(group, levels = c("Crypto", "Equity"))
  ) %>%
  ggplot(aes(x = end, y = share / 100, color = group, linetype = group)) +
  geom_line(linewidth = 0.9, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(values = c("Crypto" = "black", "Equity" = "#1f77b4"),
                     name = NULL) +
  scale_linetype_manual(values = c("Crypto" = "solid", "Equity" = "dashed"),
                        name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

Equity markets exhibit **substantially lower Financial Instability** than crypto-assets, with an average share of 5.3% against 10.1% for crypto. A pronounced peak emerges in the immediate aftermath of the Covid-19 shock in 2020, during which synchronised global equity sell-offs generated left-tail amplification across a broad cross-section of individual stocks (Ramelli & Wagner, 2020; Alfaro et al., 2020). Outside of this episode, Financial Instability in equity markets remains well below the crypto benchmark.

The gap is even more pronounced for **Financial Fragility**. The average equity share of 4.6% is less than one-third of the crypto average of 15.7%. This divergence reflects the greater susceptibility of crypto markets to herding and momentum during expansionary phases (Bouri et al., 2021; Kyriazis et al., 2020): whereas equity prices respond to fundamental revisions in cash-flow expectations during booms, crypto valuations are more directly driven by speculative demand and sentiment. In addition, equity markets exhibit larger instability than fragility, consistent with Hu (2006) who shows that stock markets are more likely to exhibit left-tail dependence than right-tail dependence.

Taken together, these results establish a **quantitative gap** in financial stability between crypto-assets and equity markets. The novelty lies not in the direction of the finding but in its grounding: the Financial Instability and Financial Fragility indices are derived from a formal test of shock transmission stability rather than from a comparison against an arbitrary benchmark — a distinction that matters because volatility-based rankings are sensitive to the choice of reference asset, whereas our classification is self-contained at the asset level.

---

# Robustness {#robustness}

## Model Specification

We run several robustness checks to confirm that our results are not dependent on model specifications. The baseline uses 18-month rolling windows, $\tau \in \{0.05, 0.50, 0.95\}$, Wald test at 10% significance, and bilateral dummies at the 5% tail threshold.

```{r robustness-code, eval=FALSE}
# ── All robustness estimations (eval=FALSE — run once via src/robust_garch_shock_ext.R)
# Outputs:
#   output/models/garch_shock_ext/robust_windows.parquet
#   output/models/garch_shock_ext/robust_tau_vec.parquet
#   output/models/garch_shock_ext/robust_alpha.parquet
#   output/models/garch_shock_ext/robust_threshold.parquet
# and corresponding PDF figures in output/figures/garch_shock_ext/robustness/
source("src/robust_garch_shock_ext.R")
```

### Alternative Window Widths

We use shorter (6 and 12 months) and longer (24 and 30 months) windows. Shorter windows are more reactive but noisier; longer windows are smoother but slower to detect crises.

```{r fig-rob-window, fig.cap="**Figure 4 — Robustness to Alternative Rolling Window Width.** The grey area spans the minimum and maximum shares across alternative window widths of 6, 12, 24, and 30 months; the solid line reports the baseline 18-month window.", fig.height=6}
rob_win <- arrow::read_parquet(sp$models$robust_windows) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, window_months) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(recode(metric,
    fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")))

band_win  <- rob_win %>%
  group_by(end, metric) %>%
  summarise(lo = min(share), hi = max(share), .groups = "drop")
base_win  <- rob_win %>% filter(window_months == 18)

ggplot(band_win, aes(x = end)) +
  geom_ribbon(aes(ymin = lo, ymax = hi), fill = "grey70", alpha = 0.5) +
  geom_line(data = base_win, aes(y = share), color = "black", linewidth = 0.9) +
  facet_wrap(~metric, ncol = 2) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

Despite the heterogeneity introduced by window choice, our results stay consistent across specifications — particularly for the Financial Instability index, while the Financial Fragility index exhibits larger timing differences but a similar overall dynamics.

### Alternative Quantile Vectors

We use a stricter specification ($\tau \in \{0.01, 0.50, 0.99\}$) and a milder specification ($\tau \in \{0.10, 0.50, 0.90\}$).

```{r fig-rob-tau, fig.cap="**Figure 5 — Robustness to Alternative Extreme Returns Quantiles.** The grey area spans the minimum and maximum shares across the strict and mild specifications; the solid line reports the baseline $\\tau \\in \\{0.05, 0.50, 0.95\\}$.", fig.height=6}
rob_tau <- arrow::read_parquet(sp$models$robust_tau_vec) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, tau_spec) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(recode(metric,
    fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")))

band_tau <- rob_tau %>%
  group_by(end, metric) %>%
  summarise(lo = min(share), hi = max(share), .groups = "drop")
base_tau <- rob_tau %>% filter(tau_spec == "baseline")

ggplot(band_tau, aes(x = end)) +
  geom_ribbon(aes(ymin = lo, ymax = hi), fill = "grey70", alpha = 0.5) +
  geom_line(data = base_tau, aes(y = share), color = "black", linewidth = 0.9) +
  facet_wrap(~metric, ncol = 2) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

The Financial Instability index is robust to the change of quantiles. The Financial Fragility index exhibits larger variation under the stricter specification, but the dynamics remains similar overall.

### Alternative Significance Levels

We report results with a tighter 5% Wald test significance level alongside the baseline 10%.

```{r fig-rob-alpha, fig.cap="**Figure 6 — Robustness to Alternative Wald Test Significance Levels.** The tighter $\\alpha = 5\\%$ threshold (grey line) alongside the baseline $\\alpha = 10\\%$ (black line).", fig.height=6}
rob_alpha <- arrow::read_parquet(sp$models$robust_alpha) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, alpha_val) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  filter(abs(alpha_val - 0.05) < 1e-9 | abs(alpha_val - 0.10) < 1e-9) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
      fi = "Financial Instability", ff = "Financial Fragility"),
      levels = c("Financial Instability", "Financial Fragility")),
    alpha_label = factor(paste0(alpha_val * 100, "%"), levels = c("5%", "10%"))
  )

ggplot(rob_alpha, aes(x = end, y = share,
                      color = alpha_label, linetype = alpha_label)) +
  geom_line(linewidth = 0.9, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(values = c("5%" = "grey60", "10%" = "black"), name = NULL) +
  scale_linetype_manual(values = c("5%" = "solid", "10%" = "solid"), name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

Consistent with using a tighter threshold, the level of Financial Instability and Financial Fragility decrease, yet the dynamics of both indices do not substantially change.

### Alternative Extreme-Shock Thresholds

We report results for $q \in \{2.5\%, 7.5\%, 10\%\}$ against the baseline $q = 5\%$.

```{r fig-rob-threshold, fig.cap="**Figure 7 — Robustness to Alternative Systematic Shock Thresholds.** The grey area spans the minimum and maximum shares across $q \\in \\{2.5\\%, 7.5\\%, 10\\%\\}$; the solid line reports the baseline $q = 5\\%$.", fig.height=6}
rob_thr <- arrow::read_parquet(sp$models$robust_threshold) %>%
  mutate(end = as.Date(end)) %>%
  group_by(end, q_threshold) %>%
  summarise(fi = mean(cont, na.rm=TRUE), ff = mean(spec, na.rm=TRUE), .groups="drop") %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(metric = factor(recode(metric,
    fi = "Financial Instability", ff = "Financial Fragility"),
    levels = c("Financial Instability", "Financial Fragility")))

band_thr <- rob_thr %>%
  group_by(end, metric) %>%
  summarise(lo = min(share), hi = max(share), .groups = "drop")
base_thr <- rob_thr %>% filter(abs(q_threshold - 0.05) < 1e-9)

ggplot(band_thr, aes(x = end)) +
  geom_ribbon(aes(ymin = lo, ymax = hi), fill = "grey70", alpha = 0.5) +
  geom_line(data = base_thr, aes(y = share), color = "black", linewidth = 0.9) +
  facet_wrap(~metric, ncol = 2) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

These alternative specifications confirm that the construction of Financial Instability and Financial Fragility is not dependent on the definition of extreme systematic shock dummies.

## Selection in the Asset Universe {#selection-bias}

A potential concern is that large-cap assets may be structurally more stable than smaller tokens (Amihud, 2002), so that restricting the sample to assets ever ranked in the top 50 could understate Financial Instability and Fragility in the broader crypto universe. To assess this, we replicate the rolling estimation on an independent sample of **150 tokens** listed on CoinMarketCap before January 2019 but absent from the baseline universe.

```{r extended-universe-code, eval=FALSE}
# ── Extended universe estimation (eval=FALSE — run once via src/tmp_extended_universe.R)
# Downloads 150 tokens via crypto2 (CoinMarketCap, no API key needed).
# Output: output/models/roll_extended_garch_shock_ext.parquet
#         output/figures/garch_shock_ext/robustness/selection_bias_extended.pdf
source("src/tmp_extended_universe.R")
```

```{r tab-selection-bias}
# ── Selection bias summary table ───────────────────────────────────────────────
roll_ext <- arrow::read_parquet(
  "output/models/roll_extended_garch_shock_ext.parquet"
) %>% mutate(end = as.Date(end))

ext_shares <- roll_ext %>%
  group_by(end) %>%
  summarise(fi = mean(cont, na.rm=TRUE)*100,
            ff = mean(spec, na.rm=TRUE)*100, .groups="drop")

data.frame(
  Index   = rep(c("Financial Instability","Financial Fragility"), 2),
  Universe = c(rep("Baseline (top-50, N=137)", 2), rep("Extended (smaller tokens, N≈146)", 2)),
  Mean    = c(round(mean(shares$fi),1), round(mean(shares$ff),1),
              round(mean(ext_shares$fi),1), round(mean(ext_shares$ff),1)),
  SD      = c(round(sd(shares$fi),1),   round(sd(shares$ff),1),
              round(sd(ext_shares$fi),1), round(sd(ext_shares$ff),1)),
  Min     = c(round(min(shares$fi),1),  round(min(shares$ff),1),
              round(min(ext_shares$fi),1), round(min(ext_shares$ff),1)),
  Max     = c(round(max(shares$fi),1),  round(max(shares$ff),1),
              round(max(ext_shares$fi),1), round(max(ext_shares$ff),1))
) %>%
  arrange(Index, Universe) %>%
  kable(caption = "**Table 2 — Financial Instability and Fragility: Baseline vs. Extended Universe**",
        col.names = c("Index","Universe","Mean (%)","Std. Dev.","Min","Max"),
        align = c("l","l","r","r","r","r")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 13) %>%
  footnote(general = "Time-series statistics computed over rolling estimation windows. Extended universe: 150 tokens listed before January 2019 and absent from the baseline sample, downloaded via CoinMarketCap (crypto2 package).")
```

The **Financial Instability index is robust**: the average share of 9.7% in the extended universe is nearly identical to the baseline 10.1%, and the episodic dynamics are preserved. The Financial Fragility share is systematically higher among smaller tokens (22.0% versus 15.8%), reflecting their greater susceptibility to speculative amplification during boom phases — a genuine compositional feature rather than a downward bias in the baseline.

```{r fig-selection-extended, fig.cap="**Figure 8 — Selection Bias: Extended Universe.** FI and FF shares for the baseline top-50 universe (black solid) and the extended sample of 150 smaller tokens (red dashed). The extended estimation uses the same rolling QR specification and GARCH-standardised shock.", fig.height=6}
bind_rows(
  shares     %>% select(end, fi, ff) %>% mutate(group = "Top-50 universe (baseline)"),
  ext_shares %>% select(end, fi, ff) %>% mutate(group = "Extended universe (smaller tokens)")
) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
      fi = "Financial Instability", ff = "Financial Fragility"),
      levels = c("Financial Instability", "Financial Fragility")),
    group  = factor(group, levels = c("Top-50 universe (baseline)",
                                      "Extended universe (smaller tokens)"))
  ) %>%
  ggplot(aes(x = end, y = share / 100, color = group, linetype = group)) +
  geom_line(linewidth = 0.85, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(
    values = c("Top-50 universe (baseline)" = "black",
               "Extended universe (smaller tokens)" = "#d62728"),
    name = NULL) +
  scale_linetype_manual(
    values = c("Top-50 universe (baseline)" = "solid",
               "Extended universe (smaller tokens)" = "dashed"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

## Endogeneity of Factor Constituents {#endogeneity}

The systematic shock $f^*_t$ is extracted from a DFM estimated on seven benchmark assets. Because these seven assets contribute directly to the construction of the shock, their estimated quantile betas may be mechanically inflated. To assess this, we recompute the shares after **excluding the seven factor constituents** from the classification. No re-estimation is required: we simply exclude BTC, ETH, XRP, BNB, ADA, TRX, and DOGE when aggregating the shares.

```{r basket-robustness-code, eval=FALSE}
# ── Factor constituent exclusion (eval=FALSE — run once via src/tmp_selection_bias_basket.R)
# Output: output/figures/garch_shock_ext/robustness/selection_bias.pdf
source("src/tmp_selection_bias_basket.R")
```

```{r fig-basket, fig.cap="**Figure 9 — Estimations excluding systematic factor constituents.** FI and FF shares after excluding the seven factor constituents from the classification sample (dashed line), alongside the baseline shares that include all assets (solid line).", fig.height=6}
BASKET <- c("BTC", "ETH", "XRP", "BNB", "ADA", "TRX", "DOGE")

restr_shares <- roll %>%
  filter(!(variable %in% BASKET)) %>%
  group_by(end) %>%
  summarise(fi = mean(cont, na.rm=TRUE)*100,
            ff = mean(spec, na.rm=TRUE)*100, .groups="drop")

bind_rows(
  shares       %>% select(end, fi, ff) %>% mutate(group = "Baseline"),
  restr_shares %>% select(end, fi, ff) %>% mutate(group = "Excl. factor constituents")
) %>%
  pivot_longer(c(fi, ff), names_to = "metric", values_to = "share") %>%
  mutate(
    metric = factor(recode(metric,
      fi = "Financial Instability", ff = "Financial Fragility"),
      levels = c("Financial Instability", "Financial Fragility")),
    group  = factor(group, levels = c("Baseline", "Excl. factor constituents"))
  ) %>%
  ggplot(aes(x = end, y = share / 100, color = group, linetype = group)) +
  geom_line(linewidth = 0.85, na.rm = TRUE) +
  facet_wrap(~metric, ncol = 2) +
  scale_color_manual(
    values = c("Baseline" = "black", "Excl. factor constituents" = "#d62728"),
    name = NULL) +
  scale_linetype_manual(
    values = c("Baseline" = "solid", "Excl. factor constituents" = "dashed"),
    name = NULL) +
  scale_x_date(date_breaks = "1 year", date_labels = "%Y",
               expand = expansion(mult = 0.02)) +
  scale_y_continuous(labels = scales::percent_format(accuracy = 1),
                     expand = expansion(mult = c(0, 0.05))) +
  labs(x = NULL, y = "Share of assets (%)") +
  theme_crypto()
```

The results confirm that the **factor constituents do not substantially drive the classification**: the FI and FF shares are nearly identical whether or not the seven basket tokens are included in the aggregation.

---

# Conclusion {#conclusion}

This paper proposes a formal, benchmark-independent measure of financial instability for crypto-asset markets. Applying the quantile regression framework of Baur & Schulze (2009) to 137 crypto-assets in 18-month rolling windows, we construct time-varying indices of Financial Instability and Financial Fragility that classify each asset according to whether its sensitivity to a GARCH-standardised systematic shock increases in the left or right tail of its return distribution. Over 2019–2026, approximately **10% of crypto-assets exhibit Financial Instability** and **16% exhibit Financial Fragility** on average. Financial Instability is episodic, concentrating around the 2022 Crypto-Winter and the SVB episode, while Financial Fragility is more persistent and peaks during the 2021 bull market — a sequencing consistent with the Minsky hypothesis that prolonged expansions accumulate financial fragilities that subsequently unwind. A comparison with 143 individual equity stocks confirms that crypto markets are structurally more unstable and fragile than equity markets, with the exception of the Covid-19 episode. A predictive portfolio exercise further validates the classification: assets satisfying financial stability conditions generate higher cumulative returns over the full sample than assets exhibiting tail amplification in either direction.

Several limitations warrant acknowledgement. The rolling window design introduces a detection lag of approximately half a window length. The systematic shock is constructed from a fixed set of seven benchmark assets; while the robustness analysis confirms that excluding these constituents does not alter the aggregate indices, alternative factor specifications remain to be explored. The minimum-coverage filter may also exclude recently listed tokens during the periods when they are most actively traded.

Several directions for future research follow naturally. First, the rolling Financial Instability index could be integrated into a macroprudential monitoring framework alongside on-chain metrics and stablecoin flows. Second, the asset-level classification opens the door to cross-sectional analyses examining whether Financial Instability is predicted by token-specific characteristics such as market depth, exchange concentration, or smart-contract exposure. Third, the framework could be extended to decentralised finance (DeFi) protocols and stablecoins. Finally, the documented divergence between Financial Instability and Fragility — one episodic and crisis-driven, the other persistent and boom-driven — raises questions about the interaction between the two dimensions over the full boom-bust cycle that a structural model would be well positioned to address.

---

# Appendix {.unnumbered}

## Summary Statistics {.unnumbered}

*(See Table 1 in Section 2.1.)*

## Crypto-Factor and Systematic Shocks {.unnumbered}

*(See Figures A1 and A2 in Section 2.2.)*

## Equity Indices {.unnumbered}

```{r tab-indices}
data.frame(
  Ticker = c("^INX","^GSPTSE","^COLCAP","^SPCLXIGPA","^BVSP","^MXX","^SPBLPGPT","^IBG",
             "^GDAX","^FCHI","^FTSE","^FTMIB","^SSMI","^AEX","^OMXS30","^OMXHPI",
             "^OMXCBPI","^DFMGI","^IRTS","^IBEX","^ATX","^BFX","^WIG","^BETI","^PX",
             "^OBXP","^ISEQ","^PSI20","^TRXFLDILP",
             "^JPXNK400","^SSEC","^HSI","^KS11","^BSESN","^NZ50C","^XU100","^AXJO",
             "^STI","^KLSE","^SET100","^JKSE","^VNI","^PSI",
             "^TASI","^JALSH","^EGX30","^NGSEINDEX",".dMIBD00000P"),
  Index = c("S&P 500 (USA)","S&P/TSX Composite (Canada)","MSCI COLCAP (Colombia)",
            "S&P/CLX IGPA (Chile)","Bovespa (Brazil)","IPC (Mexico)",
            "S&P/BVL General (Peru)","S&P/BYMA (Argentina)",
            "DAX (Germany)","CAC 40 (France)","FTSE 100 (UK)","FTSE MIB (Italy)",
            "SMI (Switzerland)","AEX (Netherlands)","OMX Stockholm 30 (Sweden)",
            "OMX Helsinki (Finland)","OMX Copenhagen (Denmark)","Dubai Fin. Market (UAE)",
            "RTS Index (Russia)","IBEX 35 (Spain)","ATX (Austria)","BEL 20 (Belgium)",
            "Warsaw General (Poland)","BET (Romania)","PX Prague (Czech Rep.)",
            "Oslo OBX (Norway)","ISEQ All Share (Ireland)","PSI-20 (Portugal)","FTSE Israel",
            "JPX-Nikkei 400 (Japan)","Shanghai Composite (China)","Hang Seng (Hong Kong)",
            "KOSPI (South Korea)","BSE Sensex (India)","S&P/NZX 50 (New Zealand)",
            "BIST 100 (Turkey)","S&P/ASX 200 (Australia)","Straits Times (Singapore)",
            "FTSE Bursa KLCI (Malaysia)","SET 100 (Thailand)","IDX Composite (Indonesia)",
            "Ho Chi Minh (Vietnam)","PSEi (Philippines)",
            "Tadawul All Share (Saudi Arabia)","FTSE/JSE All Share (South Africa)",
            "EGX 30 (Egypt)","Nigeria All Share","MSCI Bangladesh")
) %>%
  kable(caption = "**Table A1 — Equity Price Indices Used to Construct the Global Equity Factor**",
        col.names = c("Ticker","Index")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 12) %>%
  footnote(general = "The first principal component of these 49 daily price index returns is used as the global equity factor. Russia (^IRTS) is included over the full sample; observations after trading suspension are excluded.")
```

## Equity Stocks {.unnumbered}

```{r tab-stocks}
stocks_tbl <- data.frame(
  Region   = c(rep("North America",1), rep("Europe",9), rep("Asia-Pacific",5), "Total"),
  Country  = c("United States","United Kingdom","Germany","France","Netherlands",
               "Italy","Spain","Switzerland","Sweden","Denmark",
               "Japan","Hong Kong","China (US-listed)","Australia","India",""),
  Exchange = c("NYSE/NASDAQ","LSE","XETRA","Euronext Paris","Euronext Amsterdam",
               "Borsa Italiana","BME","SIX","Nasdaq Stockholm","Nasdaq Copenhagen",
               "TSE","HKEX","NYSE/NASDAQ","ASX","NSE",""),
  N        = c(60,10,10,10,5,5,4,4,4,2,10,5,3,7,4,143)
)

stocks_tbl %>%
  kable(caption = "**Table A2 — Individual Equity Stocks: Composition by Country**",
        col.names = c("Region","Country","Exchange","N"),
        align = c("l","l","l","r")) %>%
  kable_styling(bootstrap_options = c("striped","hover","condensed"),
                full_width = FALSE, font_size = 12) %>%
  row_spec(nrow(stocks_tbl), bold = TRUE) %>%
  collapse_rows(columns = 1, valign = "middle")
```

---

## Session Information {.unnumbered}

```{r session-info, class.source='fold-hide'}
sessionInfo()
```

---

*Florian Kraus — University of Bordeaux (BSE) — florian.kraus@u-bordeaux.fr*
*Compiled with R `r paste(R.version$major, R.version$minor, sep=".")` and knitr `r packageVersion("knitr")`.*
