aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJakob Odersky <jakob@odersky.com>2018-11-16 15:34:37 -0800
committerJakob Odersky <jakob@odersky.com>2018-11-16 15:34:37 -0800
commitadf0a539034a167f86200c19eec36c42f0a3e01d (patch)
tree355f2a5c43f04110c68328c0a132cdfa71664b78
parent31e077ad42cac4d1f043a6c6f949d65baf321cf6 (diff)
downloadwebsite-adf0a539034a167f86200c19eec36c42f0a3e01d.tar.gz
website-adf0a539034a167f86200c19eec36c42f0a3e01d.tar.bz2
website-adf0a539034a167f86200c19eec36c42f0a3e01d.zip
Add scalebythebay 2018 talk
-rw-r--r--talks/index.md19
-rw-r--r--talks/scala-channels.html496
-rw-r--r--talks/scala-channels.pdfbin0 -> 198721 bytes
3 files changed, 515 insertions, 0 deletions
diff --git a/talks/index.md b/talks/index.md
new file mode 100644
index 0000000..52b1804
--- /dev/null
+++ b/talks/index.md
@@ -0,0 +1,19 @@
+---
+layout: page
+title: Talks
+id: talks
+---
+
+### Channeling the Inner Complexity
+
+An essential requirement for writing programs that scale is to have
+constructs to model concurrency in an understandable, safe, and
+efficient manner. This talk presents an overview of various such
+models available in Scala, and their impact on program structure and
+complexity. It then explores a way to model concurrency with less
+complexity with an implementation of Communicating Sequential
+Processes (CSP), heavily inspired by Goroutines, scala-async and
+Clojure's core.async.
+
+[slides [pdf]](scala-channels.pdf) [[html]](scala-channels.html)
+[project escale](https://github.com/jodersky/escale)
diff --git a/talks/scala-channels.html b/talks/scala-channels.html
new file mode 100644
index 0000000..2fbbcfb
--- /dev/null
+++ b/talks/scala-channels.html
@@ -0,0 +1,496 @@
+<!DOCTYPE html>
+<html xmlns="http://www.w3.org/1999/xhtml" lang="en-US" xml:lang="en-US">
+<head>
+ <meta charset="utf-8" />
+ <meta name="generator" content="pandoc" />
+ <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" />
+ <meta name="author" content="Jakob Odersky" />
+ <meta name="dcterms.date" content="2018-11-15" />
+ <title>Channeling the Inner Complexity</title>
+ <style type="text/css">
+ code{white-space: pre-wrap;}
+ span.smallcaps{font-variant: small-caps;}
+ span.underline{text-decoration: underline;}
+ div.column{display: inline-block; vertical-align: top; width: 50%;}
+ </style>
+ <style type="text/css">
+a.sourceLine { display: inline-block; line-height: 1.25; }
+a.sourceLine { pointer-events: none; color: inherit; text-decoration: inherit; }
+a.sourceLine:empty { height: 1.2em; position: absolute; }
+.sourceCode { overflow: visible; }
+code.sourceCode { white-space: pre; position: relative; }
+div.sourceCode { margin: 1em 0; }
+pre.sourceCode { margin: 0; }
+@media screen {
+div.sourceCode { overflow: auto; }
+}
+@media print {
+code.sourceCode { white-space: pre-wrap; }
+a.sourceLine { text-indent: -1em; padding-left: 1em; }
+}
+pre.numberSource a.sourceLine
+ { position: relative; }
+pre.numberSource a.sourceLine:empty
+ { position: absolute; }
+pre.numberSource a.sourceLine::before
+ { content: attr(data-line-number);
+ position: absolute; left: -5em; text-align: right; vertical-align: baseline;
+ border: none; pointer-events: all;
+ -webkit-touch-callout: none; -webkit-user-select: none;
+ -khtml-user-select: none; -moz-user-select: none;
+ -ms-user-select: none; user-select: none;
+ padding: 0 4px; width: 4em;
+ }
+pre.numberSource { margin-left: 3em; padding-left: 4px; }
+div.sourceCode
+ { color: #cccccc; background-color: #303030; }
+@media screen {
+a.sourceLine::before { text-decoration: underline; }
+}
+code span.al { color: #ffcfaf; } /* Alert */
+code span.an { color: #7f9f7f; font-weight: bold; } /* Annotation */
+code span.at { } /* Attribute */
+code span.bn { color: #dca3a3; } /* BaseN */
+code span.bu { } /* BuiltIn */
+code span.cf { color: #f0dfaf; } /* ControlFlow */
+code span.ch { color: #dca3a3; } /* Char */
+code span.cn { color: #dca3a3; font-weight: bold; } /* Constant */
+code span.co { color: #7f9f7f; } /* Comment */
+code span.cv { color: #7f9f7f; font-weight: bold; } /* CommentVar */
+code span.do { color: #7f9f7f; } /* Documentation */
+code span.dt { color: #dfdfbf; } /* DataType */
+code span.dv { color: #dcdccc; } /* DecVal */
+code span.er { color: #c3bf9f; } /* Error */
+code span.ex { } /* Extension */
+code span.fl { color: #c0bed1; } /* Float */
+code span.fu { color: #efef8f; } /* Function */
+code span.im { } /* Import */
+code span.in { color: #7f9f7f; font-weight: bold; } /* Information */
+code span.kw { color: #f0dfaf; } /* Keyword */
+code span.op { color: #f0efd0; } /* Operator */
+code span.ot { color: #efef8f; } /* Other */
+code span.pp { color: #ffcfaf; font-weight: bold; } /* Preprocessor */
+code span.sc { color: #dca3a3; } /* SpecialChar */
+code span.ss { color: #cc9393; } /* SpecialString */
+code span.st { color: #cc9393; } /* String */
+code span.va { } /* Variable */
+code span.vs { color: #cc9393; } /* VerbatimString */
+code span.wa { color: #7f9f7f; font-weight: bold; } /* Warning */
+ </style>
+</head>
+<body>
+<header>
+<h1 class="title">Channeling the Inner Complexity</h1>
+<p class="subtitle">or, lightweight threads and channels for Scala</p>
+<p class="author">Jakob Odersky</p>
+<p class="date">2018-11-15</p>
+</header>
+<h1 id="overview">Overview</h1>
+<ul>
+<li>Basic concurrency models</li>
+<li>Futures and Promises</li>
+<li>Channels and lightweight threads</li>
+</ul>
+<h1 id="definitions">Definitions</h1>
+<ul>
+<li><p><strong>parallelism</strong>: the simultaneous execution on multiple processors of different parts of a program<a href="#fn1" class="footnote-ref" id="fnref1"><sup>1</sup></a></p></li>
+<li><p><strong>concurrency</strong>: the ability of different parts of a program to be executed out-of-order or in partial order, without affecting the final outcome<a href="#fn2" class="footnote-ref" id="fnref2"><sup>2</sup></a></p></li>
+</ul>
+<h1 id="premise">Premise</h1>
+<ul>
+<li><p><strong>scalable programs need a good concurrency model</strong></p></li>
+<li>“good”:
+<ul>
+<li>increased efficiency (take advantage of parallelism)</li>
+<li>reduced complexity</li>
+</ul></li>
+</ul>
+<hr />
+<h1 id="concurrency---threads">Concurrency - Threads</h1>
+<ul>
+<li><p>single entry point, sequence of instructions</p></li>
+<li><p>traditional way to decompose programs for parallel execution</p></li>
+<li><p>own stack and kernel resources (fairly expensive)</p></li>
+<li><p>context switches (fairly expensive)</p></li>
+<li><p>runnable on a physical processor</p></li>
+</ul>
+<hr />
+<h1 id="single-thread">Single Thread</h1>
+<div class="sourceCode" id="cb1"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb1-1" data-line-number="1"><span class="kw">def</span> <span class="fu">mkmeme</span>(imageUrl: String, text: String): Image = {</a>
+<a class="sourceLine" id="cb1-2" data-line-number="2"> <span class="kw">val</span> layer1: Image = <span class="fu">fetchUrl</span>(imageUrl) <span class="co">// network call</span></a>
+<a class="sourceLine" id="cb1-3" data-line-number="3"> <span class="kw">val</span> layer2: Image = <span class="fu">textToImage</span>(text) <span class="co">// slow</span></a>
+<a class="sourceLine" id="cb1-4" data-line-number="4"> <span class="fu">superimpose</span>(layer1, layer2) <span class="co">// need both results</span></a>
+<a class="sourceLine" id="cb1-5" data-line-number="5">}</a></code></pre></div>
+<hr />
+<h1 id="single-thread-1">Single Thread</h1>
+<ul>
+<li>concurrency unit is the whole program</li>
+</ul>
+<hr />
+<h1 id="many-threads">Many Threads</h1>
+<div class="sourceCode" id="cb2"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb2-1" data-line-number="1"><span class="kw">def</span> <span class="fu">mkmeme</span>(imageUrl: String, text: String): Image = {</a>
+<a class="sourceLine" id="cb2-2" data-line-number="2"> <span class="kw">var</span> layer1: Image = <span class="kw">null</span></a>
+<a class="sourceLine" id="cb2-3" data-line-number="3"> <span class="kw">var</span> layer2: Image = <span class="kw">null</span></a>
+<a class="sourceLine" id="cb2-4" data-line-number="4"> thread {</a>
+<a class="sourceLine" id="cb2-5" data-line-number="5"> layer1 = <span class="fu">fetchUrl</span>(imageUrl)</a>
+<a class="sourceLine" id="cb2-6" data-line-number="6"> }</a>
+<a class="sourceLine" id="cb2-7" data-line-number="7"> thread {</a>
+<a class="sourceLine" id="cb2-8" data-line-number="8"> layer2 = <span class="fu">textToImage</span>(text)</a>
+<a class="sourceLine" id="cb2-9" data-line-number="9"> }</a>
+<a class="sourceLine" id="cb2-10" data-line-number="10"> <span class="kw">while</span>(layer1 == <span class="kw">null</span> || layer2 == <span class="kw">null</span>) {</a>
+<a class="sourceLine" id="cb2-11" data-line-number="11"> <span class="co">// wait somehow</span></a>
+<a class="sourceLine" id="cb2-12" data-line-number="12"> }</a>
+<a class="sourceLine" id="cb2-13" data-line-number="13"> <span class="fu">superimpose</span>(layer1, layer2)</a>
+<a class="sourceLine" id="cb2-14" data-line-number="14">}</a></code></pre></div>
+<hr />
+<h1 id="many-threads-1">Many Threads</h1>
+<ul>
+<li><p>synchronization between threads at some point</p></li>
+<li><p>rendezvous through memory barriers (CMPXCHG)</p></li>
+<li><p>logic flow much more complex</p></li>
+<li><p>threads, blocked and running</p>
+<ul>
+<li>consume memory</li>
+<li>memory is cheap! create more threads? context switches</li>
+</ul></li>
+<li><p>threads are a low-level building block, using them efficiently is complex</p></li>
+<li><p>not available on all platforms (i.e. browser)</p></li>
+</ul>
+<hr />
+<h1 id="multiple-threads-queue-based">Multiple Threads, Queue-based</h1>
+<div class="sourceCode" id="cb3"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb3-1" data-line-number="1"><span class="kw">def</span> <span class="fu">mkmeme</span>(imageUrl: String, text: String): Image = {</a>
+<a class="sourceLine" id="cb3-2" data-line-number="2"> <span class="kw">val</span> q1 = Queue[Image]</a>
+<a class="sourceLine" id="cb3-3" data-line-number="3"> <span class="kw">val</span> q2 = Queue[Image]</a>
+<a class="sourceLine" id="cb3-4" data-line-number="4"> thread {</a>
+<a class="sourceLine" id="cb3-5" data-line-number="5"> q1.<span class="fu">put</span>(<span class="fu">fetchUrl</span>(imageUrl))</a>
+<a class="sourceLine" id="cb3-6" data-line-number="6"> }</a>
+<a class="sourceLine" id="cb3-7" data-line-number="7"> thread {</a>
+<a class="sourceLine" id="cb3-8" data-line-number="8"> q2.<span class="fu">put</span>(<span class="fu">textToImage</span>(text))</a>
+<a class="sourceLine" id="cb3-9" data-line-number="9"> }</a>
+<a class="sourceLine" id="cb3-10" data-line-number="10"> <span class="fu">superimpose</span>(q1.<span class="fu">take</span>(), q2.<span class="fu">take</span>())</a>
+<a class="sourceLine" id="cb3-11" data-line-number="11">}</a></code></pre></div>
+<h1 id="multiple-threads-queue-based-1">Multiple Threads, Queue-based</h1>
+<ul>
+<li>simpler logic flow</li>
+<li>same resource usage as plain threads</li>
+</ul>
+<hr />
+<h1 id="concurrency---callbacks">Concurrency - Callbacks</h1>
+<ul>
+<li><p>“reactive”</p></li>
+<li><p>many entrypoints</p></li>
+<li><p>register operation on event</p></li>
+<li><p>“call back” when event has happened, operation is run</p></li>
+<li>examples:
+<ul>
+<li>JavaScript</li>
+<li>libuv</li>
+<li>event loops</li>
+</ul></li>
+<li><p>in a sense, a more fundamental construct</p></li>
+</ul>
+<p>–</p>
+<h1 id="callbacks">Callbacks</h1>
+<div class="sourceCode" id="cb4"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb4-1" data-line-number="1"><span class="kw">def</span> <span class="fu">mkmeme</span>(imageUrl: String, text: String,</a>
+<a class="sourceLine" id="cb4-2" data-line-number="2"> callback: Image =&gt; Unit): Unit = {</a>
+<a class="sourceLine" id="cb4-3" data-line-number="3"> <span class="kw">var</span> layer1 = <span class="kw">null</span></a>
+<a class="sourceLine" id="cb4-4" data-line-number="4"> <span class="kw">var</span> layer2 = <span class="kw">null</span></a>
+<a class="sourceLine" id="cb4-5" data-line-number="5"> <span class="kw">def</span> <span class="fu">combine</span>() = <span class="fu">callback</span>(<span class="fu">superimpose</span>(layer1, layer2))</a>
+<a class="sourceLine" id="cb4-6" data-line-number="6"> <span class="fu">fetchUrl</span>(imageUrl, img =&gt; {</a>
+<a class="sourceLine" id="cb4-7" data-line-number="7"> layer1 = img</a>
+<a class="sourceLine" id="cb4-8" data-line-number="8"> <span class="kw">if</span> (layer2 != <span class="kw">null</span>) { <span class="co">//!\\ danger if parallelism &gt; 1</span></a>
+<a class="sourceLine" id="cb4-9" data-line-number="9"> <span class="fu">combine</span>()</a>
+<a class="sourceLine" id="cb4-10" data-line-number="10"> }</a>
+<a class="sourceLine" id="cb4-11" data-line-number="11"> })</a>
+<a class="sourceLine" id="cb4-12" data-line-number="12"> <span class="fu">textToImage</span>(text, img =&gt; {</a>
+<a class="sourceLine" id="cb4-13" data-line-number="13"> layer2 = img</a>
+<a class="sourceLine" id="cb4-14" data-line-number="14"> <span class="kw">if</span> (layer1 != <span class="kw">null</span>) {</a>
+<a class="sourceLine" id="cb4-15" data-line-number="15"> <span class="fu">combine</span>()</a>
+<a class="sourceLine" id="cb4-16" data-line-number="16"> }</a>
+<a class="sourceLine" id="cb4-17" data-line-number="17"> })</a>
+<a class="sourceLine" id="cb4-18" data-line-number="18">}</a></code></pre></div>
+<hr />
+<p><img src="" alt="callback" /><br />
+</p>
+<hr />
+<h1 id="callbacks-1">Callbacks</h1>
+<ul>
+<li><p>advantages:</p>
+<ul>
+<li>little resource overhead</li>
+<li>available on all platforms</li>
+<li>runnable on many processors</li>
+</ul></li>
+<li>disadvantage:
+<ul>
+<li>program logic quickly becomes extremely complex and scattered: <em>callback hell</em></li>
+</ul></li>
+</ul>
+<hr />
+<ul>
+<li><p>can we wrap callbacks in a more functional way?</p>
+<ul>
+<li>reduce complexity</li>
+<li>keep efficiency, and run it on ideal number of processors</li>
+</ul></li>
+</ul>
+<hr />
+<h1 id="concurrency---futures">Concurrency - Futures</h1>
+<h2 id="scala.concurrent.futurea"><code>scala.concurrent.Future[A]</code></h2>
+<ul>
+<li><p>contains an operation of result type <code>A</code></p></li>
+<li><p>transformable with <code>map</code> and <code>flatMap</code></p></li>
+<li><p>when operation is run, future completes with a result (success or failure)</p></li>
+</ul>
+<hr />
+<h1 id="future">Future</h1>
+<div class="sourceCode" id="cb5"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb5-1" data-line-number="1"><span class="kw">def</span> <span class="fu">mkmeme</span>(imageUrl: String, text: String): Future[Image] = {</a>
+<a class="sourceLine" id="cb5-2" data-line-number="2"> <span class="kw">val</span> layer1: Future[Image] = <span class="fu">fetchUrl</span>(imageUrl) </a>
+<a class="sourceLine" id="cb5-3" data-line-number="3"> <span class="kw">val</span> layer2: Future[Image] = <span class="fu">textToImage</span>(text)</a>
+<a class="sourceLine" id="cb5-4" data-line-number="4"> <span class="kw">for</span> {</a>
+<a class="sourceLine" id="cb5-5" data-line-number="5"> l1 &lt;- layer1</a>
+<a class="sourceLine" id="cb5-6" data-line-number="6"> l2 &lt;- layer2</a>
+<a class="sourceLine" id="cb5-7" data-line-number="7"> } <span class="kw">yield</span> {</a>
+<a class="sourceLine" id="cb5-8" data-line-number="8"> <span class="fu">superimpose</span>(l1, l2)</a>
+<a class="sourceLine" id="cb5-9" data-line-number="9"> }</a>
+<a class="sourceLine" id="cb5-10" data-line-number="10">}</a></code></pre></div>
+<hr />
+<h1 id="promises">Promises</h1>
+<h2 id="scala.concurren.promisea"><code>scala.concurren.Promise[A]</code></h2>
+<ul>
+<li>used to create and complete futures at the edge of the callback graph</li>
+</ul>
+<hr />
+<div class="sourceCode" id="cb6"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb6-1" data-line-number="1"><span class="co">// ScalaJS, env: browser</span></a>
+<a class="sourceLine" id="cb6-2" data-line-number="2"></a>
+<a class="sourceLine" id="cb6-3" data-line-number="3"><span class="kw">def</span> url: Future[String] = {</a>
+<a class="sourceLine" id="cb6-4" data-line-number="4"> <span class="kw">val</span> promise = Promise[String] <span class="co">// create promise</span></a>
+<a class="sourceLine" id="cb6-5" data-line-number="5"> input.<span class="fu">onsubmit</span>(_ =&gt; promise.<span class="fu">success</span>(input.<span class="fu">value</span>))</a>
+<a class="sourceLine" id="cb6-6" data-line-number="6"> promise.<span class="fu">future</span></a>
+<a class="sourceLine" id="cb6-7" data-line-number="7">}</a>
+<a class="sourceLine" id="cb6-8" data-line-number="8"></a>
+<a class="sourceLine" id="cb6-9" data-line-number="9"><span class="co">// single callback at the edge</span></a>
+<a class="sourceLine" id="cb6-10" data-line-number="10">url.<span class="fu">map</span>(fetch).<span class="fu">onComplete</span>{</a>
+<a class="sourceLine" id="cb6-11" data-line-number="11"> <span class="kw">case</span> <span class="fu">Success</span>(site) =&gt; webview.<span class="fu">value</span> = site</a>
+<a class="sourceLine" id="cb6-12" data-line-number="12"> <span class="kw">case</span> <span class="fu">Failure</span>(error) =&gt;</a>
+<a class="sourceLine" id="cb6-13" data-line-number="13"> textbox.<span class="fu">value</span> = <span class="st">&quot;oh no!&quot;</span></a>
+<a class="sourceLine" id="cb6-14" data-line-number="14"> textbox.<span class="fu">color</span> = red</a>
+<a class="sourceLine" id="cb6-15" data-line-number="15">}</a></code></pre></div>
+<h1 id="execution-contexts">Execution Contexts</h1>
+<p>Who runs a future?</p>
+<ul>
+<li>one process traverses all callbacks? no!</li>
+<li>operation “chunks” on an execution context</li>
+</ul>
+<h2 id="executioncontext"><code>ExecutionContext</code></h2>
+<ul>
+<li><p>contains graph of callbacks as chunks</p>
+<div class="sourceCode" id="cb7"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb7-1" data-line-number="1">future1.<span class="fu">flatMap</span>(f1 =&gt; <span class="fu">op1</span>(f1).<span class="fu">map</span>(<span class="fu">op2</span>(_))(ec))(ec)</a></code></pre></div></li>
+<li><p>chunks are run on a <em>ThreadPool</em></p></li>
+</ul>
+<h2 id="threadpool"><code>ThreadPool</code></h2>
+<ul>
+<li><p>(limited) group of threads</p></li>
+<li><p>every thread runs a chunk, when done takes a next chunk</p>
+<ul>
+<li>aside: <em>when done</em> <span class="math inline">←</span> this is why blocking in futures is not recomended</li>
+</ul></li>
+</ul>
+<hr />
+<h1 id="futures---composition">Futures - Composition</h1>
+<div class="sourceCode" id="cb8"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb8-1" data-line-number="1"><span class="kw">def</span> <span class="fu">lookupUser</span>(id: String): Future[Option[User]]</a>
+<a class="sourceLine" id="cb8-2" data-line-number="2"><span class="kw">def</span> <span class="fu">authorize</span>(user: User, capabilities: Set[Cap]):</a>
+<a class="sourceLine" id="cb8-3" data-line-number="3"> Future[Option[User]]</a>
+<a class="sourceLine" id="cb8-4" data-line-number="4"></a>
+<a class="sourceLine" id="cb8-5" data-line-number="5"><span class="kw">def</span> <span class="fu">authorizeduser</span>(userId: String): Future[Option[User]] = {</a>
+<a class="sourceLine" id="cb8-6" data-line-number="6"> <span class="fu">lookupUser</span>(userId).<span class="fu">flatMap</span>{</a>
+<a class="sourceLine" id="cb8-7" data-line-number="7"> <span class="kw">case</span> None =&gt; Future.<span class="fu">successful</span>(None)</a>
+<a class="sourceLine" id="cb8-8" data-line-number="8"> <span class="kw">case</span> Some(user) =&gt; <span class="fu">authorize</span>(user, Set(<span class="st">&quot;see_meme&quot;</span>))</a>
+<a class="sourceLine" id="cb8-9" data-line-number="9"> }</a>
+<a class="sourceLine" id="cb8-10" data-line-number="10">}</a></code></pre></div>
+<hr />
+<h1 id="futures---shortcomings">Futures - Shortcomings</h1>
+<ol type="1">
+<li><p>composition can be messy<a href="#fn3" class="footnote-ref" id="fnref3"><sup>3</sup></a></p></li>
+<li><p>one-shot; it is not simple to model recurrent events</p></li>
+</ol>
+<hr />
+<h1 id="solution-to-1---scala-async">Solution to 1 - Scala Async</h1>
+<ul>
+<li><p>Can we write a program that looks synchronous (single-threaded), but is split into chunks and run on a thread pool?</p></li>
+<li><p>yes, with macros!</p></li>
+<li><p>two constructs:</p>
+<ul>
+<li><code>async(a: =&gt; A): Future[A]</code> // macro</li>
+<li><code>await(f: Future[A]): A</code> // usable only in await</li>
+</ul></li>
+<li><p>installs handlers on futures to run a state machine</p></li>
+<li><p>official project of the Scala Center</p></li>
+<li><p><a href="https://github.com/scala/scala-async" class="uri">https://github.com/scala/scala-async</a></p></li>
+<li><p>see also python async</p></li>
+</ul>
+<hr />
+<div class="sourceCode" id="cb9"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb9-1" data-line-number="1"><span class="kw">import</span> scala.<span class="fu">concurrent</span>.<span class="fu">ExecutionContext</span>.<span class="fu">Implicits</span>.<span class="fu">global</span></a>
+<a class="sourceLine" id="cb9-2" data-line-number="2"><span class="kw">import</span> scala.<span class="fu">async</span>.<span class="fu">Async</span>._</a>
+<a class="sourceLine" id="cb9-3" data-line-number="3"></a>
+<a class="sourceLine" id="cb9-4" data-line-number="4"><span class="co">// looks like single-threaded code</span></a>
+<a class="sourceLine" id="cb9-5" data-line-number="5"><span class="kw">def</span> <span class="fu">mkmeme</span>(imageUrl: String, text: String): Future[Image] = </a>
+<a class="sourceLine" id="cb9-6" data-line-number="6"> async {</a>
+<a class="sourceLine" id="cb9-7" data-line-number="7"> <span class="kw">val</span> layer1 = <span class="fu">await</span>(<span class="fu">fetchUrl</span>(imageUrl))</a>
+<a class="sourceLine" id="cb9-8" data-line-number="8"> <span class="kw">val</span> layer2 = <span class="fu">await</span>(<span class="fu">textToImage</span>(text))</a>
+<a class="sourceLine" id="cb9-9" data-line-number="9"> <span class="fu">superimpose</span>(layer1, layer2)</a>
+<a class="sourceLine" id="cb9-10" data-line-number="10"> }</a></code></pre></div>
+<hr />
+<h1 id="solution-to-2---channels">Solution to 2 - Channels</h1>
+<ul>
+<li><p>futures are one-shot value</p></li>
+<li><p>queues are general useful construct for scalable programs</p>
+<ul>
+<li>separation of concerns</li>
+</ul></li>
+<li><p>as shown previously, traditional thread-based queues block</p></li>
+<li><p>can we avoid blocking, yet keep the programming model?</p></li>
+</ul>
+<h1 id="solution-to-2---channels-1">Solution to 2 - Channels</h1>
+<ul>
+<li><p>project “escale” (fr. stop, as in bus stop)</p></li>
+<li><p>inspired from Clojure’s core.async library</p></li>
+<li><p>watch Rich Hickey’s talk about it <a href="https://www.infoq.com/presentations/core-async-clojure" class="uri">https://www.infoq.com/presentations/core-async-clojure</a></p></li>
+<li>constructs:
+<ul>
+<li><code>go {block}: Future[A]</code> ~ lightweight thread</li>
+<li><code>Channel[A]</code> ~ queue</li>
+<li><code>ch.put(value: A): Future[A]</code> ~ write operation</li>
+<li><code>ch.take(): Future[A]</code> ~ read operation</li>
+<li><code>select(ch: Channel[_]*)</code></li>
+</ul></li>
+<li><p>syntax sugar</p></li>
+<li><p>form of communicating sequential processes (CSP) <span class="citation" data-cites="csp">[1]</span></p>
+<ul>
+<li>there is a formal mathematical model</li>
+</ul></li>
+<li><p>since runtime is abstracted, runs on JVM, JS and Native</p></li>
+</ul>
+<hr />
+<h1 id="escale">escale</h1>
+<div class="sourceCode" id="cb10"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb10-1" data-line-number="1"><span class="kw">import</span> scala.<span class="fu">concurrent</span>.<span class="fu">ExecutionContext</span>.<span class="fu">Implicits</span>.<span class="fu">global</span></a>
+<a class="sourceLine" id="cb10-2" data-line-number="2"><span class="kw">import</span> escale.<span class="fu">syntax</span>._</a>
+<a class="sourceLine" id="cb10-3" data-line-number="3"></a>
+<a class="sourceLine" id="cb10-4" data-line-number="4"><span class="kw">val</span> ch = chan[Int]() <span class="co">// create a channel</span></a>
+<a class="sourceLine" id="cb10-5" data-line-number="5"></a>
+<a class="sourceLine" id="cb10-6" data-line-number="6">go {</a>
+<a class="sourceLine" id="cb10-7" data-line-number="7"> ch !&lt; <span class="dv">1</span> <span class="co">// write to channel, &quot;block&quot; if no room</span></a>
+<a class="sourceLine" id="cb10-8" data-line-number="8"> <span class="fu">println</span>(<span class="st">&quot;wrote 1&quot;</span>)</a>
+<a class="sourceLine" id="cb10-9" data-line-number="9">}</a>
+<a class="sourceLine" id="cb10-10" data-line-number="10">go {</a>
+<a class="sourceLine" id="cb10-11" data-line-number="11"> ch !&lt; <span class="dv">2</span></a>
+<a class="sourceLine" id="cb10-12" data-line-number="12"> <span class="fu">println</span>(<span class="st">&quot;wrote 2&quot;</span>)</a>
+<a class="sourceLine" id="cb10-13" data-line-number="13">}</a>
+<a class="sourceLine" id="cb10-14" data-line-number="14"></a>
+<a class="sourceLine" id="cb10-15" data-line-number="15">go {</a>
+<a class="sourceLine" id="cb10-16" data-line-number="16"> <span class="kw">val</span> r: Int = !&lt;(ch) <span class="co">// read from channel</span></a>
+<a class="sourceLine" id="cb10-17" data-line-number="17"> <span class="fu">println</span>(r)</a>
+<a class="sourceLine" id="cb10-18" data-line-number="18"> <span class="fu">println</span>(!&lt;(ch))</a>
+<a class="sourceLine" id="cb10-19" data-line-number="19">}</a></code></pre></div>
+<hr />
+<h1 id="escale-1">escale</h1>
+<div class="sourceCode" id="cb11"><pre class="sourceCode scala"><code class="sourceCode scala"><a class="sourceLine" id="cb11-1" data-line-number="1"><span class="kw">import</span> escale.<span class="fu">syntax</span>._</a>
+<a class="sourceLine" id="cb11-2" data-line-number="2"></a>
+<a class="sourceLine" id="cb11-3" data-line-number="3">go {</a>
+<a class="sourceLine" id="cb11-4" data-line-number="4"> <span class="kw">val</span> Ch1 = chan[Int]() <span class="co">// create a channel</span></a>
+<a class="sourceLine" id="cb11-5" data-line-number="5"> <span class="kw">val</span> Ch2 = chan[Int]()</a>
+<a class="sourceLine" id="cb11-6" data-line-number="6"> </a>
+<a class="sourceLine" id="cb11-7" data-line-number="7"> go { Ch1 !&lt; <span class="dv">1</span> } <span class="co">// write to channel</span></a>
+<a class="sourceLine" id="cb11-8" data-line-number="8"> go { Ch2 !&lt; <span class="dv">1</span> }</a>
+<a class="sourceLine" id="cb11-9" data-line-number="9"></a>
+<a class="sourceLine" id="cb11-10" data-line-number="10"> <span class="co">// &quot;await&quot; one and only one value</span></a>
+<a class="sourceLine" id="cb11-11" data-line-number="11"> <span class="fu">select</span>(Ch1, Ch2) <span class="kw">match</span> {</a>
+<a class="sourceLine" id="cb11-12" data-line-number="12"> <span class="kw">case</span> (Ch1, value) =&gt; <span class="st">&quot;ch1 was first&quot;</span></a>
+<a class="sourceLine" id="cb11-13" data-line-number="13"> <span class="kw">case</span> (Ch2, value) =&gt; <span class="st">&quot;ch2 was first&quot;</span></a>
+<a class="sourceLine" id="cb11-14" data-line-number="14"> }</a>
+<a class="sourceLine" id="cb11-15" data-line-number="15">}</a></code></pre></div>
+<h1 id="escale---implementation">escale - Implementation</h1>
+<ul>
+<li><p>proof-of-concept</p></li>
+<li><p>https://github.com/jodersky/escale (soon)</p></li>
+<li><p>channels take care of buffering and efficient locking operations</p></li>
+<li><p>put and take return futures (select slightly more complex, but also returns a future)</p></li>
+<li><p>rely on scala-async to transform future into state machine</p></li>
+<li><p>provide syntax sugar to hide calls to <code>await</code> and alias <code>async</code></p></li>
+</ul>
+<h1 id="escale---roadmap">escale - Roadmap</h1>
+<ul>
+<li><p>channel closing and error handling</p></li>
+<li><p>deeper integration with scala async</p>
+<ul>
+<li>explore working with the state machine directly, rather than relying on double macro transformations</li>
+</ul></li>
+<li><p>select on puts</p></li>
+<li><p>buffer policies (drop first, sliding window)</p></li>
+<li><p>API improvements:</p>
+<ul>
+<li>consider replacing symbols</li>
+<li>remove wilcard <code>import escale.sytntax._</code></li>
+<li>directionality type refinements</li>
+</ul></li>
+</ul>
+<hr />
+<h1 id="summary-what-have-we-done">Summary: what have we done?</h1>
+<ul>
+<li>replaced queues and threads with conceptually lightweight queues and threads</li>
+<li>same programming model, better concurrency</li>
+<li>in a library!</li>
+</ul>
+<p><em>All problems in computer science can be solved by another level of indirection</em>.</p>
+<hr />
+<h1 id="other-approaches">Other Approaches</h1>
+<h2 id="actors">Actors</h2>
+<ul>
+<li><p>actors and CSP can be considered duals</p></li>
+<li><p>actors are named, processes are anonymous</p></li>
+<li><p>message path is anonymous, channels are named</p></li>
+<li><p>sending messages is fundamentally non-blocking, whereas (unbuffered) channels can serve as rendezvous points</p></li>
+</ul>
+<h2 id="reactive-streams">Reactive Streams</h2>
+<ul>
+<li>builds a protocol on top of actors to achieve rendezvous capabilities and backpressure</li>
+</ul>
+<hr />
+<h1 id="guidelines">Guidelines</h1>
+<p>Keep programs <em>simple</em>, it will make it <em>easier</em> for others to understand.</p>
+<ol type="1">
+<li>write synchronous logic</li>
+<li>use futures and promises with scala-async</li>
+<li>escale and other concurrency libraries</li>
+<li>…</li>
+<li>…</li>
+<li>…</li>
+<li>…</li>
+<li>…</li>
+<li>…</li>
+<li>consider callbacks</li>
+</ol>
+<hr />
+<h1 id="thank-you">Thank You!</h1>
+<ul>
+<li><p>slides: https://jakob.odersky.com/talks</p></li>
+<li><p>project: https://github.com/jodersky/escale</p></li>
+<li><p>author: <code>@jodersky</code></p></li>
+</ul>
+<hr />
+<h1 id="references" class="unnumbered">References</h1>
+<div id="refs" class="references">
+<div id="ref-csp">
+<p>[1] C. A. R. Hoare, “Communicating sequential processes,” <em>Communications of the ACM. 21 (8)</em>, pp. 666–667, 1978.</p>
+</div>
+</div>
+<section class="footnotes">
+<hr />
+<ol>
+<li id="fn1"><p>https://en.wikipedia.org/wiki/Parallelism<a href="#fnref1" class="footnote-back">↩</a></p></li>
+<li id="fn2"><p>https://en.wikipedia.org/wiki/Concurrency_(computer_science)<a href="#fnref2" class="footnote-back">↩</a></p></li>
+<li id="fn3"><p>monad transformers may help<a href="#fnref3" class="footnote-back">↩</a></p></li>
+</ol>
+</section>
+</body>
+</html>
diff --git a/talks/scala-channels.pdf b/talks/scala-channels.pdf
new file mode 100644
index 0000000..bf587aa
--- /dev/null
+++ b/talks/scala-channels.pdf
Binary files differ