<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Marko Bilal</title>
    <description>The latest articles on DEV Community by Marko Bilal (@marko911).</description>
    <link>https://dev.to/marko911</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F135014%2Ffed08c86-f414-4b2e-bb14-5d9d37f182d1.png</url>
      <title>DEV Community: Marko Bilal</title>
      <link>https://dev.to/marko911</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/marko911"/>
    <language>en</language>
    <item>
      <title>Building a React app with functional programming (Part 1)</title>
      <dc:creator>Marko Bilal</dc:creator>
      <pubDate>Wed, 13 Jan 2021 15:27:06 +0000</pubDate>
      <link>https://dev.to/marko911/building-a-react-app-with-functional-programming-part-1-40ga</link>
      <guid>https://dev.to/marko911/building-a-react-app-with-functional-programming-part-1-40ga</guid>
      <description>&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--Xq98wEPW--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/qfykaaaqmwbcmbtuqyvr.png" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--Xq98wEPW--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/qfykaaaqmwbcmbtuqyvr.png" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;Originally published at &lt;a href="https://www.markob.io/functional-programming-react/"&gt;https://www.markob.io/functional-programming-react/&lt;/a&gt; on Jan 05 2021&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;In this series we will scrap together a real world React app that allows the user to search for podcasts and audiobooks and add them to a feed that lists the latest episodes. &lt;/p&gt;

&lt;p&gt;It is intended to showcase how to use the functional programming concepts and patterns described in the fantastic &lt;a href="https://mostly-adequate.gitbooks.io/mostly-adequate-guide/content/"&gt;Professor Frisby's Mostly Adequate Guide to Functional Programming&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;We will explore how to compose functions, make the asynchronous look synchronous and leverage functional utilities from Ramda to write some succinct code.&lt;/p&gt;

&lt;p&gt;You can skip the gibberish and grab the code from the &lt;a href="https://github.com/marko911/adequate-guide-react"&gt;Github repo&lt;/a&gt;, or tinker with it live in the &lt;a href="https://codesandbox.io/s/naughty-pine-00uxu?autoresize=1&amp;amp;fontsize=14&amp;amp;hidenavigation=1&amp;amp;moduleview=1&amp;amp;theme=dark"&gt;CodeSandbox&lt;/a&gt;.&lt;/p&gt;

&lt;h3&gt;
  
  
  Core libraries
&lt;/h3&gt;

&lt;p&gt;&lt;a href="https://folktale.origamitower.com/api/v2.3.0/en/folktale.concurrency.task.html"&gt;Folktale Task&lt;/a&gt;: for the &lt;code&gt;Task&lt;/code&gt; data structure&lt;/p&gt;

&lt;p&gt;&lt;a href="https://sanctuary.js.org/"&gt;Sanctuary&lt;/a&gt;: for &lt;code&gt;Either&lt;/code&gt; and &lt;code&gt;Maybe&lt;/code&gt; data structures.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://github.com/cujojs/most"&gt;Most&lt;/a&gt;: for &lt;code&gt;Stream&lt;/code&gt; data structure (needed for consuming xml feed data)&lt;/p&gt;

&lt;p&gt;&lt;a href="https://ramdajs.com/docs/"&gt;Ramda&lt;/a&gt;: for core lambda utilities like &lt;code&gt;map&lt;/code&gt; &lt;code&gt;compose&lt;/code&gt; &lt;code&gt;reduce&lt;/code&gt; &lt;code&gt;identity&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;I like Ramda for this because their functions are not strictly curried, meaning we can pass all required arguments in any arity we like ie.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;res&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;myFunc&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;myCollection&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="c1"&gt;// or&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;res&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;myFunc&lt;/span&gt;&lt;span class="p"&gt;)(&lt;/span&gt;&lt;span class="nx"&gt;myCollection&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h3&gt;
  
  
  App requirements
&lt;/h3&gt;

&lt;h4&gt;
  
  
  Search page
&lt;/h4&gt;

&lt;p&gt;The search page is where we can look for podcasts and books using a search field and render the results in a list.&lt;br&gt;
From the rendered list of results, we can add / save the item locally so that the episodes of that subscription appear in our feed page.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Call iTunes api for 2 different queries, &lt;code&gt;podcasts&lt;/code&gt; and &lt;code&gt;audiobooks&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;Consolidate results into one list&lt;/li&gt;
&lt;li&gt;Allow user to &lt;code&gt;add&lt;/code&gt; a result to their feed, creating a subscription for feed&lt;/li&gt;
&lt;li&gt;Save subscriptions to localStorage&lt;/li&gt;
&lt;/ul&gt;
&lt;h4&gt;
  
  
  Feed page
&lt;/h4&gt;

&lt;p&gt;The feed page will simply display all the latest episodes from our subscriptions.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Get all subscriptions from &lt;code&gt;localStorage&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;Create stream from calling each subscription's feed url&lt;/li&gt;
&lt;li&gt;Consolidate streams into one and render final list&lt;/li&gt;
&lt;/ul&gt;
&lt;h2&gt;
  
  
  Search page - task and Either
&lt;/h2&gt;
&lt;h4&gt;
  
  
  iTunes API
&lt;/h4&gt;

&lt;p&gt;Because our app depends on results from the iTunes API, we will create an &lt;code&gt;api&lt;/code&gt; package first that&lt;br&gt;
will make the calls.&lt;/p&gt;

&lt;p&gt;Here is the first of 2 functions:&lt;/p&gt;

&lt;p&gt;api.js&lt;/p&gt;


&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="nx"&gt;axios&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;axios&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="nx"&gt;Task&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;task&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;folktale/concurrency/task&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;Left&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;Right&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;sanctuary-either&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;SEARCH_URL&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;https://itunes.apple.com/search&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;

&lt;span class="k"&gt;export&lt;/span&gt; &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;fetchMedia&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="nx"&gt;term&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;media&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;limit&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="mi"&gt;5&lt;/span&gt; &lt;span class="p"&gt;})&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt;
  &lt;span class="nx"&gt;term&lt;/span&gt;
    &lt;span class="p"&gt;?&lt;/span&gt; &lt;span class="nx"&gt;task&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="k"&gt;async&lt;/span&gt; &lt;span class="nx"&gt;resolver&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
        &lt;span class="k"&gt;try&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
          &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;response&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;axios&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="kd"&gt;get&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;SEARCH_URL&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
            &lt;span class="na"&gt;params&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;term&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;media&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;limit&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
          &lt;span class="p"&gt;})&lt;/span&gt;

          &lt;span class="nx"&gt;resolver&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;resolve&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;Right&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;response&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt;
        &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;catch&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;err&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
          &lt;span class="nx"&gt;resolver&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;resolve&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;Left&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;err&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt;
        &lt;span class="p"&gt;}&lt;/span&gt;
      &lt;span class="p"&gt;})&lt;/span&gt;
    &lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;Task&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="k"&gt;of&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;Left&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="kc"&gt;null&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;


&lt;p&gt;Our search function &lt;code&gt;fetchMedia&lt;/code&gt; accepts 3 arguments:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;term: the search term,&lt;/li&gt;
&lt;li&gt;media: type of media ie podcast or audiobook,&lt;/li&gt;
&lt;li&gt;limit&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;and returns a &lt;a href="https://folktale.origamitower.com/api/v2.3.0/en/folktale.concurrency.task.html"&gt;Folktale task&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;The &lt;code&gt;task&lt;/code&gt; data structure is used as a container type for asynchronous results like promises. Using this data structure allows us to &lt;a href="https://mostly-adequate.gitbooks.io/mostly-adequate-guide/content/ch08.html"&gt;containerize&lt;/a&gt;&lt;br&gt;
the api response and operate on it just like containerized values from synchronous operations.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;fetchMedia&lt;/code&gt; returns a task which makes a network call and returns our result as an &lt;code&gt;Either&lt;/code&gt; once this task is run. The &lt;code&gt;Either&lt;/code&gt; allows us to handle an error the same as a successful response. &lt;code&gt;Right&lt;/code&gt; is used to&lt;br&gt;
transfer the actual value we are looking for, and &lt;code&gt;Left&lt;/code&gt; signals that we didn't get what we were looking for and any operation on the result will be ignored&lt;br&gt;
on a &lt;code&gt;Left&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;To summarize: our result is a &lt;code&gt;task&lt;/code&gt; wrapping an &lt;code&gt;Either&lt;/code&gt; which contains our actual api response value.&lt;/p&gt;

&lt;p&gt;Now that we have our &lt;code&gt;api&lt;/code&gt; package, let's build the search page.&lt;/p&gt;

&lt;p&gt;Here is our React component:&lt;/p&gt;

&lt;p&gt;Search.jsx&lt;/p&gt;


&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;err&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;err&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="p"&gt;[]&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;success&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;res&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;path&lt;/span&gt;&lt;span class="p"&gt;([&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;data&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;results&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;])(&lt;/span&gt;&lt;span class="nx"&gt;res&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;parseResponse&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;either&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;err&lt;/span&gt;&lt;span class="p"&gt;)(&lt;/span&gt;&lt;span class="nx"&gt;success&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="nx"&gt;query&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;setQuery&lt;/span&gt;&lt;span class="p"&gt;]&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;useState&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;""&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="nx"&gt;results&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;setResults&lt;/span&gt;&lt;span class="p"&gt;]&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;useState&lt;/span&gt;&lt;span class="p"&gt;([])&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;fetchBoth&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;async&lt;/span&gt; &lt;span class="nx"&gt;e&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="nx"&gt;e&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;preventDefault&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;combineResults&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;curry&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;compose&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;flatten&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;parseResponse&lt;/span&gt;&lt;span class="p"&gt;))([&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;])&lt;/span&gt;
  &lt;span class="p"&gt;})&lt;/span&gt;

  &lt;span class="c1"&gt;// directly taken from https://mostly-adequate.gitbooks.io/mostly-adequate-guide/content/ch10.html#ships-in-bottles&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;liftA2&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;curry&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;g&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;f1&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;f2&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;f1&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;g&lt;/span&gt;&lt;span class="p"&gt;).&lt;/span&gt;&lt;span class="nx"&gt;ap&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;f2&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;})&lt;/span&gt;

  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;allTasks&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;liftA2&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
    &lt;span class="nx"&gt;combineResults&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="nx"&gt;fetchMedia&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="na"&gt;term&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;query&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;media&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;podcast&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt; &lt;span class="p"&gt;}),&lt;/span&gt;
    &lt;span class="nx"&gt;fetchMedia&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="na"&gt;term&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;query&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;media&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;audiobook&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt; &lt;span class="p"&gt;})&lt;/span&gt;
  &lt;span class="p"&gt;)&lt;/span&gt;

  &lt;span class="nx"&gt;allTasks&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;run&lt;/span&gt;&lt;span class="p"&gt;().&lt;/span&gt;&lt;span class="nx"&gt;listen&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
    &lt;span class="na"&gt;onRejected&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;val&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
      &lt;span class="nx"&gt;console&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;error&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;val&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
    &lt;span class="p"&gt;},&lt;/span&gt;
    &lt;span class="na"&gt;onResolved&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;setResults&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="p"&gt;})&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;


&lt;p&gt;So, a lot to unpack here but it is quite simple when its boiled down. I didn't include the html / jsx markup because we only need to know that &lt;code&gt;fetchBoth&lt;/code&gt; is called once user types a search term and hits fetch button.&lt;/p&gt;
&lt;h4&gt;
  
  
  Consolidating api responses
&lt;/h4&gt;

&lt;p&gt;Our search function &lt;code&gt;fetchBoth&lt;/code&gt; consolidates both audiobooks and podcasts into the results listed on the page. This is a contrived example of having to make multiple api calls in order to get a single result list. &lt;/p&gt;

&lt;p&gt;Let's start with &lt;code&gt;allTasks&lt;/code&gt;. This function lets us make two api calls in parallel and combine the results while keeping the values in&lt;br&gt;
container land.&lt;/p&gt;

&lt;p&gt;Breaking this down further, let's look at &lt;code&gt;liftA2&lt;/code&gt;.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;liftA2&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;curry&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;g&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;f1&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;f2&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;f1&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;g&lt;/span&gt;&lt;span class="p"&gt;).&lt;/span&gt;&lt;span class="nx"&gt;ap&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;f2&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This is a curried function , meaning the arguments don't need to be supplied all the same time. They can be passed in one after the other at any point, and only once all arguments are provided does it run the function body. In our case, we passed everything in all at once.&lt;/p&gt;

&lt;p&gt;Now, the &lt;code&gt;g&lt;/code&gt; argument is function that we will apply to the result of the first function &lt;code&gt;f1&lt;/code&gt;, in our case &lt;code&gt;g&lt;/code&gt; is &lt;code&gt;combineResults&lt;/code&gt; and &lt;code&gt;f1&lt;/code&gt; is the result of &lt;code&gt;fetchMedia({ term: query, media: "podcast" })&lt;/code&gt; which is a &lt;code&gt;task&lt;/code&gt;. &lt;/p&gt;

&lt;p&gt;Since &lt;code&gt;task&lt;/code&gt; is a functor, it has a &lt;code&gt;map&lt;/code&gt; method and so our &lt;code&gt;liftA2&lt;/code&gt; works.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;combineResults&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;curry&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;compose&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;flatten&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;parseResponse&lt;/span&gt;&lt;span class="p"&gt;))([&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;])&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Remember, once we map over a &lt;code&gt;task&lt;/code&gt;, we are extracting the containerized value, so &lt;code&gt;f1.map(combineResults)&lt;/code&gt; returns us a &lt;code&gt;task&lt;/code&gt; but with the containerized value of &lt;code&gt;f1&lt;/code&gt; passed into &lt;code&gt;combineResults&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;The containerized value is a &lt;code&gt;Left&lt;/code&gt; or a &lt;code&gt;Right&lt;/code&gt;, and &lt;code&gt;combinedResults&lt;/code&gt; is waiting for the 2nd argument before it will run. &lt;code&gt;.ap&lt;/code&gt; method of &lt;code&gt;task&lt;/code&gt; allows us to apply the results of &lt;code&gt;f2&lt;/code&gt; (which again will be a &lt;code&gt;Left&lt;/code&gt;&lt;br&gt;
or a &lt;code&gt;Right&lt;/code&gt;) as the 2nd argument to &lt;code&gt;combineResults&lt;/code&gt; and returns a task. In the end , we get to work with those containerized values without popping them out manually.&lt;/p&gt;

&lt;p&gt;And since &lt;code&gt;combineResults&lt;/code&gt; is a curried function, it gives us the ability to fire&lt;br&gt;
off both api calls (the &lt;code&gt;fetchMedia&lt;/code&gt; functions) in parallel without having to coordinate and wait for results of each and then pass them into &lt;code&gt;combineResults&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;So in the end, all we are doing is calling &lt;code&gt;combineResults&lt;/code&gt; with the values from inside the two &lt;code&gt;tasks&lt;/code&gt; (which are &lt;code&gt;Either&lt;/code&gt;) returned by the &lt;code&gt;fetchMedia&lt;/code&gt; functions.&lt;/p&gt;

&lt;p&gt;Once both api calls return, the body of the &lt;code&gt;combineResults&lt;/code&gt; function runs. Note: as you will see below, we must run the &lt;code&gt;task&lt;/code&gt; that is returned&lt;br&gt;
from &lt;code&gt;liftA2&lt;/code&gt; in order to kick off the entire chain of events.&lt;/p&gt;
&lt;h4&gt;
  
  
  Parsing and data transformation with map, compose and either
&lt;/h4&gt;


&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight jsx"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;err&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;err&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="p"&gt;[]&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;success&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;res&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;path&lt;/span&gt;&lt;span class="p"&gt;([&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;data&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;results&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;])(&lt;/span&gt;&lt;span class="nx"&gt;res&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;parseResponse&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;either&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;err&lt;/span&gt;&lt;span class="p"&gt;)(&lt;/span&gt;&lt;span class="nx"&gt;success&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;combineResults&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;curry&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;compose&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;flatten&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;parseResponse&lt;/span&gt;&lt;span class="p"&gt;))([&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;])&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;


&lt;p&gt;Here we can showcase why an &lt;code&gt;Either&lt;/code&gt; is useful and some Ramda goodies. In the implementation above, we simply grabbed the results of the calls and passed them to &lt;code&gt;parseResponse&lt;/code&gt; which extracts the value. But we can also do something like transforming the data before we extract it. &lt;/p&gt;

&lt;p&gt;Our response has a &lt;code&gt;url&lt;/code&gt; field and we want to &lt;a href="https://ramdajs.com/docs/#toUpper"&gt;&lt;code&gt;toUpper&lt;/code&gt; &lt;/a&gt;it before parsing.&lt;/p&gt;

&lt;p&gt;Just add that part to the &lt;a href="https://ramdajs.com/docs/#toUpper"&gt; &lt;code&gt;compose&lt;/code&gt;&lt;/a&gt; :&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight jsx"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;upperUrl&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;over&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;lensPath&lt;/span&gt;&lt;span class="p"&gt;([&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;config&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;url&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;]),&lt;/span&gt; &lt;span class="nx"&gt;toUpper&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;combineResults&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;curry&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;compose&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
    &lt;span class="nx"&gt;flatten&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;parseResponse&lt;/span&gt;&lt;span class="p"&gt;),&lt;/span&gt;
    &lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;upperUrl&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt;
  &lt;span class="p"&gt;)([&lt;/span&gt;&lt;span class="nx"&gt;pods&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;audiobooks&lt;/span&gt;&lt;span class="p"&gt;])&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The &lt;code&gt;upperUrl&lt;/code&gt; function takes a response object , and applies the &lt;a href="https://ramdajs.com/docs/#toUpper"&gt;&lt;code&gt;toUpper&lt;/code&gt; &lt;/a&gt; function to the "config.url" path of the object and returns a new object with the url uppercased.&lt;/p&gt;

&lt;p&gt;We must &lt;a href="https://ramdajs.com/docs/#map"&gt;&lt;code&gt;map&lt;/code&gt; &lt;/a&gt; twice because remember, first map will unwrap theunderlying value, the 2nd map operates on it. The beauty here is that if either &lt;code&gt;pods&lt;/code&gt; or &lt;code&gt;audiobooks&lt;/code&gt; returns a &lt;code&gt;Left&lt;/code&gt;, we don't have to&lt;br&gt;
have special case code for that, the Either takes care of it for us and the uppercase transformation simply doesn't run on a &lt;code&gt;Left&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;compose&lt;/code&gt;, &lt;code&gt;map&lt;/code&gt;and&lt;code&gt;flatten&lt;/code&gt;are all utilities from Ramda.&lt;a href="https://ramdajs.com/docs/#compose"&gt;&lt;code&gt;compose&lt;/code&gt;&lt;/a&gt;takes a series of functions and returns a function that passes it's argument to the supplied list of functions from right to left. So we pass the array&lt;code&gt;[pods, audiobooks]&lt;/code&gt;to&lt;code&gt;map(parseResponse)&lt;/code&gt;and then&lt;code&gt;flatten&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;This is where we get to pop out our values from their containers and use them in the render. Some folks might even call this function from render to keep the impure part&lt;br&gt;
at the render step.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;const parseResponse = either(err)(success)&lt;/code&gt; uses &lt;code&gt;either&lt;/code&gt; which is an import from &lt;code&gt;sancturay&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;This &lt;code&gt;either&lt;/code&gt; function extracts the containerized value.&lt;br&gt;
The first argument you supply, in our case the &lt;code&gt;err&lt;/code&gt; function, is the result returned if the &lt;code&gt;Either&lt;/code&gt; is a &lt;code&gt;Left&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;The second argument is returned if it is a &lt;code&gt;Right&lt;/code&gt;. If the underlying &lt;code&gt;task&lt;/code&gt; returns a &lt;code&gt;Left(err)&lt;/code&gt;, we are ignoring the &lt;code&gt;err&lt;/code&gt; argument and just returning an empty array. &lt;/p&gt;

&lt;p&gt;We can instead throw the error, &lt;code&gt;console.log&lt;/code&gt; it , or pass it to a bug catch api for logging. If the &lt;code&gt;task&lt;/code&gt; wraps a &lt;code&gt;Right&lt;/code&gt; , then the &lt;code&gt;success&lt;/code&gt; function runs with the argument as the&lt;br&gt;
value the &lt;code&gt;Right&lt;/code&gt; wraps.&lt;/p&gt;

&lt;p&gt;As you can see, our &lt;code&gt;success&lt;/code&gt; function takes the value the &lt;code&gt;Right&lt;/code&gt; provides and returns the object path "data.results".&lt;br&gt;
See &lt;a href="https://ramdajs.com/docs/#path"&gt;path from Ramda&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;And the final part is that we &lt;code&gt;flatten&lt;/code&gt; the results because we will have each api response as a list of results so it ends up a nested array.&lt;/p&gt;

&lt;p&gt;In order to actually kickoff the task, we must run&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight jsx"&gt;&lt;code&gt;&lt;span class="nx"&gt;otherTasks&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;run&lt;/span&gt;&lt;span class="p"&gt;().&lt;/span&gt;&lt;span class="nx"&gt;listen&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;onRejected&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;val&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="nx"&gt;console&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;error&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;val&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;},&lt;/span&gt;
  &lt;span class="na"&gt;onResolved&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;setResults&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;If you notice, we do not have any async/await logic inside our React component. The resolve/reject logic is whatever we specify in the &lt;code&gt;.listen()&lt;/code&gt; method.&lt;/p&gt;

&lt;p&gt;This ends our Search page, in the next part we tackle saving subscriptions and displaying a feed of latest episodes.&lt;/p&gt;

</description>
      <category>react</category>
      <category>functional</category>
      <category>ramda</category>
      <category>sanctuary</category>
    </item>
  </channel>
</rss>
