Skip to content

Instantly share code, notes, and snippets.

@josevalim
Created May 20, 2017 18:18
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save josevalim/86bcbc9b84c4b9fe543eca4f0d69bdef to your computer and use it in GitHub Desktop.
Save josevalim/86bcbc9b84c4b9fe543eca4f0d69bdef to your computer and use it in GitHub Desktop.
# Run it inside plug directory with: mix run --no-halt sample.exs
defmodule MyRouter do
use Plug.Router
use Plug.Debugger
plug :reload
plug :match
plug :dispatch
defp reload(conn, _) do
IEx.Helpers.r(Plug.Debugger)
conn
end
get "/oops" do
Keyword.pop(:foo, 2)
conn
end
end
{:ok, _} = Plug.Adapters.Cowboy.http MyRouter, port: 4000
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>FunctionClauseError at GET /oops</title>
<meta name="viewport" content="width=device-width">
<style>/*! normalize.css v4.2.0 | MIT License | github.com/necolas/normalize.css */html{font-family:sans-serif;line-height:1.15;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,details,figcaption,figure,footer,header,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block}audio:not([controls]){display:none;height:0}progress{vertical-align:baseline}template,[hidden]{display:none}a{background-color:transparent;-webkit-text-decoration-skip:objects}a:active,a:hover{outline-width:0}abbr[title]{border-bottom:none;text-decoration:underline;text-decoration:underline dotted}b,strong{font-weight:inherit}b,strong{font-weight:bolder}dfn{font-style:italic}h1{font-size:2em;margin:0.67em 0}mark{background-color:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-0.25em}sup{top:-0.5em}img{border-style:none}svg:not(:root){overflow:hidden}code,kbd,pre,samp{font-family:monospace, monospace;font-size:1em}figure{margin:1em 40px}hr{box-sizing:content-box;height:0;overflow:visible}button,input,optgroup,select,textarea{font:inherit;margin:0}optgroup{font-weight:bold}button,input{overflow:visible}button,select{text-transform:none}button,html [type="button"],[type="reset"],[type="submit"]{-webkit-appearance:button}button::-moz-focus-inner,[type="button"]::-moz-focus-inner,[type="reset"]::-moz-focus-inner,[type="submit"]::-moz-focus-inner{border-style:none;padding:0}button:-moz-focusring,[type="button"]:-moz-focusring,[type="reset"]:-moz-focusring,[type="submit"]:-moz-focusring{outline:1px dotted ButtonText}fieldset{border:1px solid #c0c0c0;margin:0 2px;padding:0.35em 0.625em 0.75em}legend{box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}textarea{overflow:auto}[type="checkbox"],[type="radio"]{box-sizing:border-box;padding:0}[type="number"]::-webkit-inner-spin-button,[type="number"]::-webkit-outer-spin-button{height:auto}[type="search"]{-webkit-appearance:textfield;outline-offset:-2px}[type="search"]::-webkit-search-cancel-button,[type="search"]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-input-placeholder{color:inherit;opacity:0.54}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}</style>
<style>
html, body, td, input {
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", "Roboto", "Oxygen", "Ubuntu", "Cantarell", "Fira Sans", "Droid Sans", "Helvetica Neue", sans-serif;
}
* {
box-sizing: border-box;
}
html {
font-size: 15px;
line-height: 1.6;
background: #fff;
color: #304050;
}
@media (max-width: 768px) {
html {
font-size: 14px;
}
}
@media (max-width: 480px) {
html {
font-size: 13px;
}
}
button:focus,
summary:focus {
outline: 0;
}
summary {
cursor: pointer;
}
pre {
font-family: menlo, consolas, monospace;
overflow: auto;
max-width: 100%;
}
.top-details {
padding: 48px;
background: #f9f9fa;
}
.top-details,
.conn-info {
padding: 48px;
}
@media (max-width: 768px) {
.top-details,
.conn-info {
padding: 32px;
}
}
@media (max-width: 480px) {
.top-details,
.conn-info {
padding: 16px;
}
}
/*
* Exception logo
*/
.exception-logo {
position: absolute;
right: 48px;
top: 48px;
pointer-events: none;
width: 100%;
}
.exception-logo:before {
content: '';
display: block;
height: 64px;
width: 100%;
background-size: auto 100%;
background-image: url("");
background-position: right 0;
background-repeat: no-repeat;
margin-bottom: 16px;
}
@media (max-width: 768px) {
.exception-logo {
position: static;
}
.exception-logo:before {
height: 32px;
background-position: left 0;
}
}
@media (max-width: 480px) {
.exception-logo {
display: none;
}
}
/*
* Exception info
*/
/* Compensate for logo placement */
@media (min-width: 769px) {
.exception-info {
max-width: 90%;
}
}
.exception-info > .struct,
.exception-info > .title,
.exception-info > .detail {
margin: 0;
padding: 0;
}
.exception-info > .struct {
font-size: 1em;
font-weight: 700;
color: #4e2a8e;
}
.exception-info > .struct > small {
font-size: 1em;
color: #a0b0c0;
font-weight: 400;
}
/* Hide path until hover */
.exception-info > .struct > .path {
opacity: 0;
-webkit-transition: opacity 100ms linear;
transition: opacity 100ms linear;
}
.exception-info:hover > .struct > .path {
opacity: 1;
}
.exception-info > .title {
font-size: 2.0736em;
line-height: 1.4;
font-weight: 300;
color: #4e2a8e;
}
@media (max-width: 768px) {
.exception-info > .title {
font-size: 1.7490062499999994em;
}
}
@media (max-width: 480px) {
.exception-info > .title {
font-size: 1.4641000000000004em;
}
}
.exception-info > .detail {
margin-top: 1.3em;
white-space: pre;
}
/*
* Code explorer
*/
.code-explorer {
margin: 32px 0 0 0;
}
@media (max-width: 768px) {
.code-explorer {
margin-top: 16px;
}
}
.code-explorer:after {
content: '';
display: table;
clear: both;
zoom: 1;
}
.code-explorer > .code-snippets {
float: left;
width: 45%;
}
.code-explorer > .stack-trace {
float: right;
width: 55%;
padding-left: 32px;
}
/* Collapse to single-column */
@media (max-width: 960px) {
.code-explorer > .code-snippets {
float: none;
width: auto;
margin-bottom: 16px;
}
.code-explorer > .stack-trace {
float: none;
width: auto;
padding-left: 0;
}
}
/*
* Snippets
*/
.code-snippets {
}
/*
* Frame info:
* Holds the code (code-block) and more
*/
.frame-info {
background: white;
box-shadow:
0 1px 3px rgba(80, 100, 140, .1),
0 8px 15px rgba(80, 100, 140, .05);
}
.frame-info > .meta,
.frame-info > .file {
padding: 12px 16px;
white-space: no-wrap;
font-size: 0.8333333333333334em;
}
@media (max-width: 480px) {
.frame-info > .meta,
.frame-info > .file {
padding: 6px 16px;
font-size: 0.9090909090909091em;
}
}
.frame-info > .file > a {
text-decoration: none;
color: #304050;
font-weight: 700;
}
.frame-info > .code {
border-top: solid 1px #eee;
border-bottom: solid 1px #eee;
}
/* Hiding */
.frame-info {
display: none;
}
.frame-info.-active {
display: block;
}
.frame-info > details.meta {
border-top: solid 1px #eee;
padding: 0;
}
.frame-info > details.meta > summary {
padding: 12px 16px;
}
/*
* Frame details
*/
.frame-mfa {
color: #a0b0c0;
}
.frame-mfa > .app {
color: #4e2a8e;
font-weight: 700;
}
.frame-mfa > .app:after {
content: '·';
margin: 0 .2em;
}
/*
* Code block:
* The `pre` that holds the code
*/
.code-block {
margin: 0;
padding: 12px 0;
font-size: .8em;
line-height: 1.4;
white-space: normal;
}
.code-block > .line {
white-space: pre;
display: block;
padding: 0 16px;
}
/* Line highlight */
.code-block > .line.-highlight {
background-color: #f0f4fa;
-webkit-animation: line-highlight 750ms linear;
animation: line-highlight 750ms linear;
}
@-webkit-keyframes line-highlight {
0% { background-color: #f0f4fa; }
25% { background-color: #ffe5e5; }
50% { background-color: #f0f4fa; }
75% { background-color: #ffe5e5; }
}
@keyframes line-highlight {
0% { background-color: #f0f4fa; }
25% { background-color: #ffe5e5; }
50% { background-color: #f0f4fa; }
75% { background-color: #ffe5e5; }
}
.code-block > .line > .ln {
color: #a0b0c0;
margin-right: 1.5em;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
.code-block > .line > .code {
font-family: menlo, consolas, monospace;
}
/*
* Empty code
*/
.code-block-empty {
text-align: center;
color: #a0b0c0;
padding-top: 48px;
padding-bottom: 48px;
}
/*
* Stack trace heading
*/
.stack-trace-heading {
padding-top: 8px;
}
.stack-trace-heading:after {
content: '';
display: block;
clear: both;
zoom: 1;
border-bottom: solid 1px #eee;
padding-top: 12px;
margin-bottom: 16px;
}
.stack-trace-heading > h3 {
display: none;
}
.stack-trace-heading > label {
display: block;
padding-left: 8px;
line-height: 1.9;
font-size: 0.8333333333333334em;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
.stack-trace-heading > label > input {
margin-right: .3em;
}
@media (max-width: 480px) {
.stack-trace-heading > label {
font-size: 0.9090909090909091em;
}
}
/*
* Stack trace
*/
.stack-trace-list,
.stack-trace-list > li {
margin: 0;
padding: 0;
list-style-type: none;
}
.stack-trace-list > li > .stack-trace-item.-all {
display: none;
}
.stack-trace-list.-show-all > li > .stack-trace-item.-all {
display: block;
}
/*
* Stack trace item:
* The clickable line to inspect a stack trace
*/
.stack-trace-item {
font-size: 0.8333333333333334em;
display: block;
width: 100%;
border: 0;
margin: 0;
padding: 4px 8px;
background: transparent;
cursor: pointer;
text-align: left;
overflow: hidden;
white-space: nowrap;
}
.stack-trace-item:hover,
.stack-trace-item:focus {
background-color: rgba(80, 100, 140, 0.05);
}
.stack-trace-item,
.stack-trace-item:active {
color: #304050;
}
.stack-trace-item:active {
background-color: rgba(80, 100, 140, 0.1);
}
.stack-trace-item.-active {
background-color: white;
}
/* Circle */
.stack-trace-item > .left:before {
content: '';
display: inline-block;
width: 8px;
height: 8px;
background: #a0b0c0;
border-radius: 50%;
margin-right: 8px;
}
.stack-trace-item.-app > .left:before {
background: #4e2a8e;
opacity: 1;
}
.stack-trace-item.-app > .left > .app {
display: none;
}
.stack-trace-item > .left {
float: left;
max-width: 55%;
}
.stack-trace-item > .info {
color: #607080;
float: right;
max-width: 45%;
}
.stack-trace-item > .left,
.stack-trace-item > .info {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.stack-trace-item > .left > .filename > .line {
color: #a0b0c0;
}
/* App name */
.stack-trace-item > .left > .app {
color: #a0b0c0;
}
.stack-trace-item > .left > .app:after {
content: '·';
margin: 0 .2em;
}
/*
* Code as a blockquote:
* Like `pre` but with wrapping
*/
.code-quote {
font-family: menlo, consolas, monospace;
font-size: 0.8333333333333334em;
margin: 0;
overflow: auto;
max-width: 100%;
word-wrap: break-word;
white-space: normal;
}
.code-quote.-padded {
padding: 0 16px 16px 16px;
}
/*
* Conn info
*/
.conn-info {
border-top: solid 1px #eee;
}
/*
* Conn details
*/
.conn-details {
}
.conn-details + .conn-details {
margin-top: 16px;
}
.conn-details > summary {
}
.conn-details > dl {
display: block;
overflow: hidden;
margin: 0;
padding: 4px 0;
border-bottom: solid 1px #eee;
white-space: nowrap;
text-overflow: ellipsis;
}
.conn-details > dl:first-of-type {
margin-top: 16px;
border-top: solid 1px #eee;
}
/* Term */
.conn-details > dl > dt {
width: 20%;
float: left;
font-size: 0.8333333333333334em;
color: #607080;
overflow: hidden;
text-overflow: ellipsis;
position: relative;
top: -1px; /* Compensate for font metrics */
}
/* Definition */
.conn-details > dl > dd {
width: 80%;
float: left;
}
@media (max-width: 480px) {
.conn-details > dl > dt {
font-size: 0.9090909090909091em;
}
}
</style>
</head>
<body>
<div class="top-details">
<aside class="exception-logo"></aside>
<header class="exception-info">
<h5 class="struct">
FunctionClauseError
<small>at GET</small>
<small class="path">/oops</small>
</h5>
<h1 class="title">no function clause matching in Keyword.pop/3</h1>
</header>
<div class="code-explorer">
<div class="code-snippets">
<div class="frame-info" data-index="0" role="stack-trace-details">
<div class="file">
<a href="">lib/keyword.ex</a>
</div>
<pre class="code code-block"><span class="line "><span class="ln">894</span><span class="code"> iex&gt; Keyword.pop([a: 1, a: 2], :a)</span></span>
<span class="line "><span class="ln">895</span><span class="code"> {1, []}</span></span>
<span class="line "><span class="ln">896</span><span class="code"></span></span>
<span class="line "><span class="ln">897</span><span class="code"> &quot;&quot;&quot;</span></span>
<span class="line "><span class="ln">898</span><span class="code"> @spec pop(t, key, value) :: {value, t}</span></span>
<span class="line -highlight"><span class="ln">899</span><span class="code"> def pop(keywords, key, default \\ nil) when is_list(keywords) do</span></span>
<span class="line "><span class="ln">900</span><span class="code"> case fetch(keywords, key) do</span></span>
<span class="line "><span class="ln">901</span><span class="code"> {:ok, value} -&gt;</span></span>
<span class="line "><span class="ln">902</span><span class="code"> {value, delete(keywords, key)}</span></span>
<span class="line "><span class="ln">903</span><span class="code"> :error -&gt;</span></span>
<span class="line "><span class="ln">904</span><span class="code"> {default, keywords}</span></span>
</pre>
<div class="meta">
<div class="frame-mfa">
<span class="app">elixir</span>
Keyword.pop/3
<a href="https://hexdocs.pm/elixir/1.5.0/Keyword.html#pop/3">(docs)</a>
</div>
</div>
<details class="meta">
<summary>Called with 3 arguments</summary>
<blockquote class="code-quote -padded">
<ol>
<li>:foo</li>
<li>2</li>
<li>nil</li>
</ol>
</blockquote>
</details>
<details class="meta">
<summary>Available clauses (1 out of 1)</summary>
<blockquote class="code-quote -padded">
<ul>
<li>def pop(keywords, key, default) when is_list(keywords)</li>
</ul>
</blockquote>
</details>
</div>
<div class="frame-info" data-index="1" role="stack-trace-details">
<div class="file">
<a href="">foo.exs</a>
</div>
<pre class="code code-block"><span class="line "><span class="ln">10</span><span class="code"> IEx.Helpers.r(Plug.Debugger)</span></span>
<span class="line "><span class="ln">11</span><span class="code"> conn</span></span>
<span class="line "><span class="ln">12</span><span class="code"> end</span></span>
<span class="line "><span class="ln">13</span><span class="code"></span></span>
<span class="line "><span class="ln">14</span><span class="code"> get &quot;/oops&quot; do</span></span>
<span class="line -highlight"><span class="ln">15</span><span class="code"> Keyword.pop(:foo, 2)</span></span>
<span class="line "><span class="ln">16</span><span class="code"> conn</span></span>
<span class="line "><span class="ln">17</span><span class="code"> end</span></span>
<span class="line "><span class="ln">18</span><span class="code">end</span></span>
<span class="line "><span class="ln">19</span><span class="code"></span></span>
<span class="line "><span class="ln">20</span><span class="code">{:ok, _} = Plug.Adapters.Cowboy.http MyRouter, port: 4000</span></span>
</pre>
<div class="meta">
<div class="frame-mfa">
anonymous fn/1 in MyRouter.do_match/4
</div>
</div>
</div>
<div class="frame-info" data-index="2" role="stack-trace-details">
<div class="file">
<a href="">foo.exs</a>
</div>
<pre class="code code-block"><span class="line -highlight"><span class="ln">1</span><span class="code">defmodule MyRouter do</span></span>
<span class="line "><span class="ln">2</span><span class="code"> use Plug.Router</span></span>
<span class="line "><span class="ln">3</span><span class="code"> use Plug.Debugger</span></span>
<span class="line "><span class="ln">4</span><span class="code"></span></span>
<span class="line "><span class="ln">5</span><span class="code"> plug :reload</span></span>
<span class="line "><span class="ln">6</span><span class="code"> plug :match</span></span>
</pre>
<div class="meta">
<div class="frame-mfa">
MyRouter.plug_builder_call/2
</div>
</div>
</div>
<div class="frame-info" data-index="3" role="stack-trace-details">
<div class="file">
<a href="">lib/plug/debugger.ex</a>
</div>
<pre class="code code-block"><span class="line "><span class="ln">95</span><span class="code"> quote location: :keep do</span></span>
<span class="line "><span class="ln">96</span><span class="code"> defoverridable [call: 2]</span></span>
<span class="line "><span class="ln">97</span><span class="code"></span></span>
<span class="line "><span class="ln">98</span><span class="code"> def call(conn, opts) do</span></span>
<span class="line "><span class="ln">99</span><span class="code"> try do</span></span>
<span class="line -highlight"><span class="ln">100</span><span class="code"> super(conn, opts)</span></span>
<span class="line "><span class="ln">101</span><span class="code"> catch</span></span>
<span class="line "><span class="ln">102</span><span class="code"> kind, reason -&gt;</span></span>
<span class="line "><span class="ln">103</span><span class="code"> Plug.Debugger.__catch__(conn, kind, reason, @plug_debugger)</span></span>
<span class="line "><span class="ln">104</span><span class="code"> end</span></span>
<span class="line "><span class="ln">105</span><span class="code"> end</span></span>
</pre>
<div class="meta">
<div class="frame-mfa">
MyRouter.call/2
</div>
</div>
</div>
<div class="frame-info" data-index="4" role="stack-trace-details">
<div class="file">
<a href="">lib/plug/adapters/cowboy/handler.ex</a>
</div>
<pre class="code code-block"><span class="line "><span class="ln">10</span><span class="code"> def upgrade(req, env, __MODULE__, {transport, plug, opts}) do</span></span>
<span class="line "><span class="ln">11</span><span class="code"> conn = @connection.conn(req, transport)</span></span>
<span class="line "><span class="ln">12</span><span class="code"> try do</span></span>
<span class="line "><span class="ln">13</span><span class="code"> %{adapter: {@connection, req}} =</span></span>
<span class="line "><span class="ln">14</span><span class="code"> conn</span></span>
<span class="line -highlight"><span class="ln">15</span><span class="code"> |&gt; plug.call(opts)</span></span>
<span class="line "><span class="ln">16</span><span class="code"> |&gt; maybe_send(plug)</span></span>
<span class="line "><span class="ln">17</span><span class="code"></span></span>
<span class="line "><span class="ln">18</span><span class="code"> {:ok, req, [{:result, :ok} | env]}</span></span>
<span class="line "><span class="ln">19</span><span class="code"> catch</span></span>
<span class="line "><span class="ln">20</span><span class="code"> :error, value -&gt;</span></span>
</pre>
<div class="meta">
<div class="frame-mfa">
<span class="app">plug</span>
Plug.Adapters.Cowboy.Handler.upgrade/4
</div>
</div>
</div>
<div class="frame-info" data-index="5" role="stack-trace-details">
<div class="file">
<a href="">src/cowboy_protocol.erl</a>
</div>
<pre class="code code-block"><span class="line "><span class="ln">437</span><span class="code">-spec execute(cowboy_req:req(), #state{}, cowboy_middleware:env(), [module()])</span></span>
<span class="line "><span class="ln">438</span><span class="code"> -&gt; ok.</span></span>
<span class="line "><span class="ln">439</span><span class="code">execute(Req, State, Env, []) -&gt;</span></span>
<span class="line "><span class="ln">440</span><span class="code"> next_request(Req, State, get_value(result, Env, ok));</span></span>
<span class="line "><span class="ln">441</span><span class="code">execute(Req, State, Env, [Middleware|Tail]) -&gt;</span></span>
<span class="line -highlight"><span class="ln">442</span><span class="code"> case Middleware:execute(Req, Env) of</span></span>
<span class="line "><span class="ln">443</span><span class="code"> {ok, Req2, Env2} -&gt;</span></span>
<span class="line "><span class="ln">444</span><span class="code"> execute(Req2, State, Env2, Tail);</span></span>
<span class="line "><span class="ln">445</span><span class="code"> {suspend, Module, Function, Args} -&gt;</span></span>
<span class="line "><span class="ln">446</span><span class="code"> erlang:hibernate(?MODULE, resume,</span></span>
<span class="line "><span class="ln">447</span><span class="code"> [State, Env, Tail, Module, Function, Args]);</span></span>
</pre>
<div class="meta">
<div class="frame-mfa">
<span class="app">cowboy</span>
:cowboy_protocol.execute/4
</div>
</div>
</div>
</div>
<div class="stack-trace">
<div class="stack-trace-heading">
<label><input type="checkbox" role="show-all-toggle"> Show all frames</label>
</div>
<ul class="stack-trace-list" role="stack-trace-list">
<li>
<button class="stack-trace-item -all" role="stack-trace-item" data-index="0">
<span class="left">
<span class="app">elixir</span>
<span class="filename">
lib/keyword.ex<span class="line">:899</span>
</span>
</span>
<span class="info">Keyword.pop/3</span>
</button>
</li>
<li>
<button class="stack-trace-item -all" role="stack-trace-item" data-index="1">
<span class="left">
<span class="filename">
foo.exs<span class="line">:15</span>
</span>
</span>
<span class="info">anonymous fn/1 in MyRouter.do_match/4</span>
</button>
</li>
<li>
<button class="stack-trace-item -all" role="stack-trace-item" data-index="2">
<span class="left">
<span class="filename">
foo.exs<span class="line">:1</span>
</span>
</span>
<span class="info">MyRouter.plug_builder_call/2</span>
</button>
</li>
<li>
<button class="stack-trace-item -all" role="stack-trace-item" data-index="3">
<span class="left">
<span class="filename">
lib/plug/debugger.ex<span class="line">:100</span>
</span>
</span>
<span class="info">MyRouter.call/2</span>
</button>
</li>
<li>
<button class="stack-trace-item -all" role="stack-trace-item" data-index="4">
<span class="left">
<span class="app">plug</span>
<span class="filename">
lib/plug/adapters/cowboy/handler.ex<span class="line">:15</span>
</span>
</span>
<span class="info">Plug.Adapters.Cowboy.Handler.upgrade/4</span>
</button>
</li>
<li>
<button class="stack-trace-item -all" role="stack-trace-item" data-index="5">
<span class="left">
<span class="app">cowboy</span>
<span class="filename">
src/cowboy_protocol.erl<span class="line">:442</span>
</span>
</span>
<span class="info">:cowboy_protocol.execute/4</span>
</button>
</li>
</ul>
</div>
</div>
</div>
<div class="conn-info">
<details class="conn-details">
<summary>Request info</summary>
<dl>
<dt>URI:</dt>
<dd class="code-quote">http://localhost:4000/oops</dd>
</dl>
<dl>
<dt>Query string:</dt>
<dd class="code-quote"></dd>
</dl>
<dl>
<dt>Peer:</dt>
<dd class="code-quote">127.0.0.1:57238</dd>
</dl>
</details>
<details class="conn-details">
<summary>Headers</summary>
<dl>
<dt>accept</dt>
<dd class="code-quote">text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8</dd>
</dl>
<dl>
<dt>accept-encoding</dt>
<dd class="code-quote">gzip, deflate, sdch, br</dd>
</dl>
<dl>
<dt>accept-language</dt>
<dd class="code-quote">en-US,en;q=0.8,pt;q=0.6</dd>
</dl>
<dl>
<dt>cache-control</dt>
<dd class="code-quote">max-age=0</dd>
</dl>
<dl>
<dt>connection</dt>
<dd class="code-quote">keep-alive</dd>
</dl>
<dl>
<dt>cookie</dt>
<dd class="code-quote">_ga=GA1.1.567573718.1494673430</dd>
</dl>
<dl>
<dt>dnt</dt>
<dd class="code-quote">1</dd>
</dl>
<dl>
<dt>host</dt>
<dd class="code-quote">localhost:4000</dd>
</dl>
<dl>
<dt>upgrade-insecure-requests</dt>
<dd class="code-quote">1</dd>
</dl>
<dl>
<dt>user-agent</dt>
<dd class="code-quote">Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.110 Safari/537.36</dd>
</dl>
</details>
</div>
<script>(function () {
var $items = document.querySelectorAll('[role~="stack-trace-item"]')
var $toggle = document.querySelector('[role~="show-all-toggle"]')
var $list = document.querySelector('[role~="stack-trace-list"]')
each($items, function ($item) {
on($item, 'click', itemOnclick)
})
on($toggle, 'click', toggleOnclick)
// Auto-check "show all" if there are no app frames.
if (document.querySelectorAll('[role~="stack-trace-list"] .-app').length === 0) {
$toggle.checked = true
toggleOnclick.call($toggle)
}
function toggleOnclick () {
if (this.checked) {
addClass($list, '-show-all')
} else {
removeClass($list, '-show-all')
}
}
function itemOnclick () {
var idx = this.getAttribute('data-index')
var $detail = document.querySelector('[role~="stack-trace-details"].-active')
if ($detail) removeClass($detail, '-active')
$detail = document.querySelector('[role~="stack-trace-details"][data-index="' + idx + '"]')
if ($detail) addClass($detail, '-active')
var $item = document.querySelector('[role~="stack-trace-item"].-active')
if ($item) removeClass($item, '-active')
$item = document.querySelector('[role~="stack-trace-item"][data-index="' + idx + '"]')
if ($item) addClass($item, '-active')
}
var $first =
document.querySelector('[role~="stack-trace-item"].-app:first-of-type') ||
document.querySelector('[role~="stack-trace-item"]:first-of-type')
if ($first) itemOnclick.call($first)
/*
* Helpers
*/
function each (list, fn) {
for (var i = 0, len = list.length; i < len; i++) {
var item = list[i]
fn(item)
}
}
function addClass (el, className) {
if (el.classList) {
el.classList.add(className)
} else {
el.className += ' ' + className
}
}
function removeClass (el, className) {
if (el.classList) {
el.classList.remove(className)
} else {
var expr = new RegExp('(^|\\b)' + className.split(' ').join('|') + '(\\b|$)', 'gi')
el.className = el.className.replace(expr, ' ')
}
}
function on (el, event, handler) {
if (el.addEventListener) {
el.addEventListener(event, handler)
} else {
el.attachEvent('on' + event, function () {
handler.call(el)
})
}
}
}())</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment