<?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: Chidume Nnamdi</title>
    <description>The latest articles on DEV Community by Chidume Nnamdi (@philipszdavido).</description>
    <link>https://dev.to/philipszdavido</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%2F37427%2F0a07e22b-235d-4545-ac8f-15b39e445b7e.jpeg</url>
      <title>DEV Community: Chidume Nnamdi</title>
      <link>https://dev.to/philipszdavido</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/philipszdavido"/>
    <language>en</language>
    <item>
      <title>SOLID: Dependency Inversion Principle in Angular</title>
      <dc:creator>Chidume Nnamdi</dc:creator>
      <pubDate>Thu, 19 Sep 2019 11:00:49 +0000</pubDate>
      <link>https://dev.to/philipszdavido/solid-dependency-inversion-principle-in-angular-1h3g</link>
      <guid>https://dev.to/philipszdavido/solid-dependency-inversion-principle-in-angular-1h3g</guid>
      <description>&lt;p&gt;&lt;a href="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fcdn-images-1.medium.com%2Fmax%2F2560%2F1%2A_0Qt3IrDReUYROo387OIVg.jpeg" class="article-body-image-wrapper"&gt;&lt;img src="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fcdn-images-1.medium.com%2Fmax%2F2560%2F1%2A_0Qt3IrDReUYROo387OIVg.jpeg"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Post originally published at &lt;a href="https://blog.bitsrc.io/solid-the-dependency-inversion-principle-in-angular-6e4b9c484960?source=---------6------------------" rel="noopener noreferrer"&gt;blog.bitsrc.io&lt;/a&gt;&lt;/p&gt;
&lt;h1&gt;
  
  
  &lt;em&gt;A. HIGH-LEVEL MODULES SHOULD NOT DEPEND UPON LOW-LEVEL MODULES. BOTH SHOULD DEPEND UPON ABSTRACTIONS.&lt;/em&gt;
&lt;/h1&gt;
&lt;h1&gt;
  
  
  &lt;em&gt;B. ABSTRACTIONS SHOULD NOT DEPEND UPON DETAILS. DETAILS SHOULD DEPEND UPON ABSTRACTIONS.&lt;/em&gt;
&lt;/h1&gt;
&lt;/blockquote&gt;

&lt;p&gt;This principle states that classes and modules should depend on abstractions not on concretions.&lt;/p&gt;

&lt;blockquote&gt;
&lt;h1&gt;
  
  
  &lt;em&gt;The Dependency Inversion Principle (DIP) tells us that the most flexible systems are those in which source code dependencies refer only to abstractions, not to concretions.&lt;/em&gt;
&lt;/h1&gt;
&lt;/blockquote&gt;

&lt;h3&gt;
  
  
  Tip: Use Bit to get the most out of your SOLID Angular project
&lt;/h3&gt;

&lt;p&gt;SOLID code is modular and reusable. With &lt;a href="https://bit.dev" rel="noopener noreferrer"&gt;**Bit&lt;/a&gt;, &lt;strong&gt;you can easily **share and organize your reusable components.&lt;/strong&gt; Let your team see what you’ve been working on, install and reuse your components across projects, and even collaborate together on individual components. &lt;a href="https://bit.dev" rel="noopener noreferrer"&gt;Give it a try&lt;/a&gt;.&lt;br&gt;
&lt;a href="https://bit.dev" rel="noopener noreferrer"&gt;&lt;strong&gt;Share reusable code components as a team · Bit&lt;/strong&gt;&lt;br&gt;
*Easily share reusable components between projects and applications to build faster as a team. Collaborate to develop…*bit.dev&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;What are Abstractions?&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Abstractions are Interfaces. Interfaces define what implementing Classes must-have. If we have an Interface Meal:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;interface Meal {
    type: string
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;This holds information on what type of meal is being served; Breakfast, Lunch or Dinner. Implementing classes like BreakFastMeal, LunchMeal and DinnerMeal must have the type property:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;class BreakFastMeal implements Meal {
    type: string = "Breakfast"
}

class LunchMeal implements Meal {
    type: string = "Lunch"
}

class DinnerMeal implements Meal {
    type: string = "Dinner"
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;So, you see Interface gives the information on what properties and methods the class that implements it must have. An Interface is called an Abstraction because it is focused on the characteristic of a Class rather than the Class as a whole group of characteristics.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;What are Concretions?&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Concretions are Classes. They are the opposite of Abstractions, they contain the full implementation of their characteristics. Above we stated that the interface Meal is an abstraction, then the classes that implemented it, DinnerMeal, BreakfastMeal and LunchMeal are the concretions, because they contain the full implementation of the Meal interface. Meal has a characteristic type and said it should be a string type, then the BreakfastMeal came and said the type is "Breakfast", LunchMeal said the type is "Lunch".&lt;/p&gt;

&lt;p&gt;The DIP says that if we depend on Concretions, it will make our class or module tightly coupled to the detail. The coupling between components results in a rigid system that is hard to change, and one that fails when changes are introduced.&lt;/p&gt;

&lt;h2&gt;
  
  
  Example: Copier
&lt;/h2&gt;

&lt;p&gt;Let’s use an example to demonstrate the effects of using the DIP. Let’s say we have a program that gets input from a disk and copies the content to a flash drive.&lt;/p&gt;

&lt;p&gt;The program would read a character from the disk and pass it to the module that will write it to the flash drive.&lt;/p&gt;

&lt;p&gt;The source will look like this:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;function Copy() {
    let bytes = []
    while(ReadFromDisk(bytes))
        WriteToFlashDrv(bytes)
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Yes, that’s a work well done, but this system is rigid, not flexible. The system is restricted to only reading from a disk and writing to a flash drive. What happens when the client wants to read from a disk and write to a network? We will see ourself adding an if statement to support the new addition&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;function Copy(to) {
    let bytes = []
    while(ReadFromDisk(bytes))
        if(to == To.Net)
            WriteToNet(bytes)
        else
            WriteToFlashDrv(bytes)
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;See we &lt;em&gt;touched&lt;/em&gt; the code, which shouldn’t be so. As time goes on, and more and more devices must participate in the copy program, the Copy function will be littered with if/else statements and will be dependent upon many lower-level modules. It will eventually become rigid and fragile.&lt;/p&gt;

&lt;p&gt;To make the Copy function reusable and less-fragile, we will implement interfaces Writer and Reader so that any place we want to read from will implement the Reader interface and any place we want to write to will implement the Write interface:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;interface Writer {
    write(bytes)
}

interface Reader {
    read(bytes)
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Now, our disk reader would implement the Reader interface:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;class DiskReader implements Reader {
    read(bytes) {
        //.. implementation here
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;then, network writer and flash drive writer would both implement the Writer interface:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;class Network implements Writer {
    write(bytes) {
        // network implementation here
    }
}

class FlashDrv implements Writer {
    write(bytes) {
        // flash drive implementation
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The Copy function would be like this:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;function Copy(to) {
    let bytes = []
    while(ReadFromDisk(bytes))
        if(to == To.Net)
            WriteToNet(bytes)
        else
            WriteToFlashDrv(bytes)
}


|
|
v

function Copy(writer: Writer, reader: Reader) {
    let bytes = []
    while(reader.read(bytes))
        writer.write(bytes)
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;See, our Copy has been shortened to a few codes. The Copy function now depends on interfaces, all it knows is that the Reader will have a read method it would call to write bytes and a Reader with a read method where it would get bytes to write, it doesn’t concern how to get the data, it is the responsibility of the class implementing the Writer.&lt;/p&gt;

&lt;p&gt;This makes the Copy function highly re-usable and less-fragile. We can pass any Reader or Writer to the Copy function, all it cares:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;// read from disk and write to flash drive
Copy(FlasDrvWriter, DiskReader)

// read from flash and write to disk
Copy(DiskWriter, FlasDrvReader)

// read from disk and write to network ie. uploading
Copy(NetworkWriter, DiskReader)

// read from network and write to disk ie. downloading
Copy(DiskWriter, NetworkReader)
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;
&lt;h2&gt;
  
  
  Example: Nodejs Console class
&lt;/h2&gt;

&lt;p&gt;The Nodejs Console class is an example of a real-world app that obeys the DIP. The Console class produces output, yeah majorly used to output to a terminal, but it can be used to output to other media like:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;file&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;network&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;When we do console.log(“Nnamdi”)&lt;/p&gt;

&lt;p&gt;Nnamdi is printed to the screen, we can channel the output to another place like we outlined above.&lt;/p&gt;

&lt;p&gt;Looking at the Console class&lt;/p&gt;
&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;function Console(stdout, stderr) {
    this.stdout = stdout
    this.stderr = stderr ? stderr : stdout
}

Console.prototype.log = function (whatToWrite) {
    this.write(whatToWrite, this.stdout)
}

Console.prototype.error = function (whatToWrite) {
    this.write(whatToWrite, this.stderr)
}

Console.prototype.write = function (whatToWrite, stream) {
    stream.write(whatToWrite)
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;It accepts a stdout and stderr which are streams, they are generic, the stream can be a terminal or file or anywhere like network stream. stdout is where to write out, the stderr is where it write any error. The console object we have globally has already been initialized with stream set to be written to terminal:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;global.console = new Console(process.stdout, process.stderr)
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The stdout and stderr are interfaces that have the write method, all that Console knows is to call the write method of the stdout and stderr.&lt;/p&gt;

&lt;p&gt;The Console depends on abstracts stdout and stderr, it is left for the user to supply the output stream and must have the write method.&lt;/p&gt;

&lt;p&gt;To make the Console class write to file we simply create a file stream:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;const fsStream = fs.createWritestream('./log.log')
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Our file is log.log, we created a writable stream to it using fs's createWriteStream API.&lt;/p&gt;

&lt;p&gt;We can create another stream we can log our error report:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;const errfsStream = fs.createWritestream('./error.log')
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;We can now pass the two streams to the Console class:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;const log = new Console(fsStream, errfsStream)
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;When we call log.log("logging an input to ./log.log"), it won't print it to the screen, rather it will write the message to the ./log.log file in your directory.&lt;/p&gt;

&lt;p&gt;Simple, the Console does not have to have a long chain of if/else statement to support any stream.&lt;/p&gt;

&lt;h2&gt;
  
  
  Angular
&lt;/h2&gt;

&lt;p&gt;Coming to Angular how do we obey the DIP?&lt;/p&gt;

&lt;p&gt;Let’s say we have a billing app that lists peoples license and calculates their fees, our app may look like this:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;@Component({
    template: `
        &amp;lt;div&amp;gt;
            &amp;lt;h3&amp;gt;License&amp;lt;/h3&amp;gt;
            &amp;lt;div *ngFor="let p of people"&amp;gt;
                &amp;lt;p&amp;gt;Name: {{p.name}}&amp;lt;/p&amp;gt;
                &amp;lt;p&amp;gt;License: {{p.licenseType}}&amp;lt;/p&amp;gt;
                &amp;lt;p&amp;gt;Fee: {{calculateFee(p)}}&amp;lt;/p&amp;gt;
            &amp;lt;/div&amp;gt;
        &amp;lt;/div&amp;gt;    
    `
})
export class App {
    people = [
        {
            name: 'Nnamdi',
            licenseType: 'personal'
        },
        {
            name: 'John',
            licenseType: 'buisness'
        },
        // ...
    ]

    constructor(private licenseService: LicenseService) {}

    calculateLicenseFee(p) {
        return this.licenseService.calculateFee(p)        
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;We have a Service that calculates the fees based on the license:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;@Injectable()
export class LicenseService {
    calculateFee(data) {
        if(data.licenseType == "personal")
             //... calculate fee based on "personal" licnese type
        else
         //... calculate fee based on "buisness" licnese type
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;This Service class violates the DIP, when another license type is introduced we will see ourself adding another if statement branch to support the new addition:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;@Injectable()
export class LicenseService {
    calculateFee(data) {
        if(data.licenseType == "personal")
             //... calculate fee based on "personal" licnese type
        else if(data.licenseType == "new license type")
            //... calculate the fee based on "new license type" license type
        else
            //... calculate fee based on "buisness" licnese type
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;To make it obey the DIP, we will create a License interface:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;interface License {
    calcFee():
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Then we can have classes that implement it like:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;class PersonalLicense implements License {
    calcFee() {
        //... calculate fee based on "personal" licnese type
    }
    // ... other methods and properties
}

class BuisnessLicense implements License {
    calcFee() {
        //... calculate fee based on "buisness" licnese type
    }
    // ... other methods and properties
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Then, we will refactor the LicenseService class:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;@Injectable()
export class LicenseService {
    calculateFee(data: License) {
        return data.calcFee()
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;It accepts data which is a License type, now we can send any license type to the LicenseService#calculateFee, it does not care about the type of license, it just knows that the data is a License type and calls its calcFee method. It is left for the class that implements the License interface to provide its license fee calculation in the calcFee method.&lt;/p&gt;

&lt;p&gt;Angular itself also obeys the DIP, in its source. For example in the Pipe concept.&lt;/p&gt;

&lt;h2&gt;
  
  
  Pipe
&lt;/h2&gt;

&lt;p&gt;Pipe is used to transform data without affecting the source. In array, we transform data like:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;mapping&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;filtering&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;sorting&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;splicing&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;slicing&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;substring &lt;em&gt;wink emoji here&lt;/em&gt;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;etc&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;All these transform data based on the implementation.&lt;/p&gt;

&lt;p&gt;In Angular templates, if we did not have the Pipe interface, we would have classes that transform data pipe like the Number, Date, JSON or custom pipe, etc. Angular would have its implementation of Pipe like this:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;pipe(pipeInstance) {
    if (pipeInstance.type == 'number')
        // transform number
    if(pipeInstance.type == 'date')
        // transform date
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The list would expand if Angular adds new pipes and it would be more problematic to support custom pipes.&lt;/p&gt;

&lt;p&gt;So to Angular created a PipeTransform interface that all pipes would implement:&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;interface PipeTransform {
    transform(data: any)
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Now any Pipe would implement the interface and provides its piping function/algorithm in the transform method.&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;@Pipe(...)
class NumberPipe implements PipeTransform {
    transform(num: any) {
        // ...
    }
}

@Pipe(...)
class DatePipe implements PipeTransform {
    transform(date: any) {
        // ...
    }
}

@Pipe(...)
class JsonPipe implements PipeTransform {
    transform(jsonData: any) {
        // ...
    }
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;Now, Angular would call the transform without bothering about the type of the Pipe&lt;/p&gt;


&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;function pipe(pipeInstance: PipeTransform, data: any) {&lt;br&gt;
    return pipeInstance.transform(data)&lt;br&gt;
}&lt;br&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;
&lt;h2&gt;
&lt;br&gt;
  &lt;br&gt;
  &lt;br&gt;
  Conclusion&lt;br&gt;
&lt;/h2&gt;

&lt;p&gt;We saw in this post how DIP makes us write reusable and maintainable code in Angular and in OOP as a whole.&lt;/p&gt;

&lt;p&gt;In &lt;strong&gt;&lt;em&gt;Engineering Notebook columns for The C++ Report&lt;/em&gt;&lt;/strong&gt; in &lt;em&gt;The Dependency Inversion Principle&lt;/em&gt; column, it says:&lt;/p&gt;

&lt;p&gt;A piece of software that fulfills its requirements and yet exhibits any or all of the following three traits has a bad design.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;&lt;p&gt;It is hard to change because every change affects too many other parts of the system. (Rigidity)&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;When you make a change, unexpected parts of the system break. (Fragility)&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;It is hard to reuse in another application because it cannot be disentangled from the current application. (Immobility)&lt;/p&gt;&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;If you have any question regarding this or anything I should add, correct or remove, feel free to comment, email or DM me&lt;/p&gt;

&lt;p&gt;Thanks !!!&lt;/p&gt;

&lt;h2&gt;
  
  
  Learn More
&lt;/h2&gt;

&lt;p&gt;&lt;a href="https://blog.bitsrc.io/how-to-share-angular-components-between-project-and-apps-5eb0600d99d2" rel="noopener noreferrer"&gt;&lt;strong&gt;How to Share Angular Components Between Projects and Apps&lt;/strong&gt;&lt;br&gt;
*Share and collaborate on NG components across projects, to build your apps faster.*blog.bitsrc.io&lt;/a&gt;&lt;br&gt;
&lt;a href="https://blog.bitsrc.io/announcing-bit-with-angular-public-beta-578cbb173690" rel="noopener noreferrer"&gt;&lt;strong&gt;Announcing Bit with Angular Public Beta&lt;/strong&gt;&lt;br&gt;
*Special thanks to the awesome Angular team for working together on making this happen 👐*blog.bitsrc.io&lt;/a&gt;&lt;/p&gt;

</description>
      <category>javascript</category>
      <category>node</category>
      <category>solid</category>
      <category>angular</category>
    </item>
  </channel>
</rss>
