<?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>Forem: Na-Dev</title>
    <description>The latest articles on Forem by Na-Dev (@na_dev_22af7e4ce79).</description>
    <link>https://forem.com/na_dev_22af7e4ce79</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%2F1703359%2Fc17af3bc-82a4-4d44-8efb-e5b07e11b59f.png</url>
      <title>Forem: Na-Dev</title>
      <link>https://forem.com/na_dev_22af7e4ce79</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/na_dev_22af7e4ce79"/>
    <language>en</language>
    <item>
      <title>Authorizing Sport: Firebase + Permit.io</title>
      <dc:creator>Na-Dev</dc:creator>
      <pubDate>Thu, 15 May 2025 20:02:05 +0000</pubDate>
      <link>https://forem.com/na_dev_22af7e4ce79/authorizing-sport-firebase-permitio-404a</link>
      <guid>https://forem.com/na_dev_22af7e4ce79/authorizing-sport-firebase-permitio-404a</guid>
      <description>&lt;p&gt;&lt;em&gt;This is a submission for the &lt;a href="https://dev.to/challenges/permit_io"&gt;Permit.io Authorization Challenge&lt;/a&gt;: Permissions Redefined&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;NB: This article was posted after the submission deadline for the Permit.io dev challenge.&lt;/em&gt;&lt;/p&gt;

&lt;h1&gt;
  
  
  Little League – Fine-Grained Authorization for Casual 3-a-Side Football Matches
&lt;/h1&gt;

&lt;p&gt;Finding players for a casual game of soccer can be a headache — organizing who's in, who's out, and when to start often takes more time than the actual match. I built &lt;strong&gt;Little League&lt;/strong&gt; to fix that.&lt;/p&gt;

&lt;p&gt;This simple web app allows players to queue into 3-a-side soccer matches. The first three players are assigned to Team A, and the next three to Team B, all handled automatically. Each team is given a randomly generated team name using the unique-names-generator package to add some personality and fun to each match.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fpex6gmq06z6r66js8hqj.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fpex6gmq06z6r66js8hqj.png" alt="Image description" width="479" height="528"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Behind the scenes, &lt;strong&gt;Permit.io&lt;/strong&gt; handles all fine-grained authorization to keep role-based access secure and easy to manage.&lt;/p&gt;

&lt;h2&gt;
  
  
  🔧 Tech Stack
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Frontend&lt;/strong&gt;: Angular&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Backend&lt;/strong&gt;: Firebase Functions&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Database&lt;/strong&gt;: Firebase Firestore&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Authentication&lt;/strong&gt;: Firebase Auth (Gmail login)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Authorization&lt;/strong&gt;: Permit.io SDK (Node.js)&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  🎯 Project Purpose
&lt;/h2&gt;

&lt;p&gt;This app demonstrates how to implement &lt;strong&gt;role-based authorization&lt;/strong&gt; using Permit.io in a real-time booking scenario with Firebase handling identity and storage.&lt;/p&gt;

&lt;h2&gt;
  
  
  🔐 Firebase + Permit.io: A Dual I/O Model
&lt;/h2&gt;

&lt;p&gt;This project uses:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Firebase&lt;/strong&gt; for authentication and real-time match state&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Permit.io&lt;/strong&gt; for enforcing role-based permissions on the frontend and backend&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Every feature tied to a user’s role is pre-checked using Permit.io’s API and SDK.&lt;/p&gt;

&lt;h2&gt;
  
  
  🚀 Getting Started
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Head to the homepage and &lt;strong&gt;log in via Gmail&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;You’ll be automatically assigned a random display name and the &lt;strong&gt;"Player"&lt;/strong&gt; role&lt;/li&gt;
&lt;li&gt;After logging in, you’ll be taken to the &lt;strong&gt;Matches&lt;/strong&gt; page&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fxxbovleao3j0ywhf7nxy.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fxxbovleao3j0ywhf7nxy.png" alt="Image description" width="461" height="644"&gt;&lt;/a&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  👥 User Roles &amp;amp; Permissions
&lt;/h2&gt;

&lt;h3&gt;
  
  
  🧑‍💼 Club Manager
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;Can &lt;strong&gt;create matches&lt;/strong&gt; via the "+New" button&lt;/li&gt;
&lt;li&gt;Can &lt;strong&gt;view player profiles&lt;/strong&gt; via a nav item&lt;/li&gt;
&lt;li&gt;On the Matches page: sees a &lt;strong&gt;View&lt;/strong&gt; button but no booking controls&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fauygy4dnfdo70q1r0n5c.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fauygy4dnfdo70q1r0n5c.png" alt="Image description" width="800" height="92"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fdr5qhbvyr22ihyatxeju.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fdr5qhbvyr22ihyatxeju.png" alt="Image description" width="800" height="225"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fyl9928gjb8bj0rhjkd0w.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fyl9928gjb8bj0rhjkd0w.png" alt="Image description" width="800" height="168"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  ⚖️ Referee
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;Cannot create matches or see player profiles&lt;/li&gt;
&lt;li&gt;Can &lt;strong&gt;view matches&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;On the &lt;strong&gt;View Match&lt;/strong&gt; page, has access to the &lt;strong&gt;Match Controller&lt;/strong&gt; where they can:

&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;Modify the score&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Update match status&lt;/strong&gt; (e.g., Not Started, In Progress, Completed)&lt;/li&gt;
&lt;/ul&gt;


&lt;/li&gt;

&lt;/ul&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fkajomnggxxg95uqyztph.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fkajomnggxxg95uqyztph.png" alt="Image description" width="800" height="176"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fouhdpvflwd83ulk1rpl5.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fouhdpvflwd83ulk1rpl5.png" alt="Image description" width="800" height="238"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  👟 Player
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;Can &lt;strong&gt;book in&lt;/strong&gt; and &lt;strong&gt;book out&lt;/strong&gt; of matches&lt;/li&gt;
&lt;li&gt;Automatically assigned to Team A or Team B depending on queue position&lt;/li&gt;
&lt;li&gt;Can see other players on a match&lt;/li&gt;
&lt;li&gt;Cannot create matches or access the match controller&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fo5et7fk28xa5bsrdhupr.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fo5et7fk28xa5bsrdhupr.png" alt="Image description" width="800" height="202"&gt;&lt;/a&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  🔄 How Authorization Works
&lt;/h2&gt;

&lt;p&gt;Every feature above is permission-guarded using &lt;strong&gt;Permit.io’s Node SDK&lt;/strong&gt;. For example:&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;allowed&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;permit&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;check&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;userId&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;create&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;match&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="o"&gt;!&lt;/span&gt;&lt;span class="nx"&gt;allowed&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;throw&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Error&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Unauthorized&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;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  🧪 Live Demo
&lt;/h2&gt;

&lt;p&gt;Curious to see how it all works in action?&lt;/p&gt;

&lt;p&gt;👉 &lt;a href="https://dev-to-permit-io.web.app/" rel="noopener noreferrer"&gt;Launch the Live Demo&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;You can log in using Gmail — you'll automatically be assigned the &lt;strong&gt;Player&lt;/strong&gt; role.&lt;/p&gt;

&lt;p&gt;To test other roles:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Demo credentials for &lt;strong&gt;Manager&lt;/strong&gt; and &lt;strong&gt;Referee&lt;/strong&gt; are available on the homepage&lt;/li&gt;
&lt;li&gt;After logging in, explore the Matches page and available actions depending on your role&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Try:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Booking into a match as a player&lt;/li&gt;
&lt;li&gt;Creating a match as a manager&lt;/li&gt;
&lt;li&gt;Controlling the match state as a referee&lt;/li&gt;
&lt;/ul&gt;

</description>
      <category>devchallenge</category>
      <category>permitchallenge</category>
      <category>webdev</category>
      <category>security</category>
    </item>
  </channel>
</rss>
