Data

Driven

Front-End

// Whoami

Alex Albu

Software Engineer @FlowX.ai

Trainer @JSLeague

// Why

// Inspiration

Back-end architecture

Micro-service architecture

Layered Architecture

// Inspiration

Software Design Techniques

reactive programming

low coupling

high cohesion

// Solution

The architecture is composed of layers each communicating only with its neighbours.

We are not going to care about stack, framework, libraries or specific implementation details.

Changes should only affect the layers above (children).

Data Handling

Responsibility to handle data from outside sources, transforming it and making it accessible to our app

// Solution

Business Logic

It holds the actual logic of how things work.

Rules, constraints, logical thinking goes into here.

// Solution

UI Components

These are reusable, generic, abstract, bits of UI that we can leverage to fast prototype the interface.

// Solution

Views

User interaction gets handled here but without the logic. Just handling events, triggers, rendering, ui actions. All logic gets delegated to the business logic.

// Solution

// Data Driven Front-end

// Data Driven Front-end

// Data Driven Front-end

// Data Driven Front-end

// Demo

Movies App

// Demo -- Data Sources

// Demo -- Models

// Demo -- Models

// rating.enum.ts
export enum Rating {
  G,
  PG,
  PG13,
  R,
  M,
}
// genre.enum.ts
export enum Genre {
  Comedy,
  Romance
}
// movie.entity.ts
import { Genre } from "@movieApp/entities";
import { Rating } from "@movieApp/entities";

export type Movie = {
  title: string;
  year: number;
  plot: string;
  posterUrl: string;
  genre: Genre[];
  rating: Rating;
};
// someplace.ts
import { Movie } from "@movieApp/entities";

// Demo -- APIs

// Demo -- APIs

// movies.services.ts
export enum APIService {
  OMDb = "https://omdb.example.com/api",
  IMDB = "https://imdb.example.com/api",
  Netflix = "https://netflix.example.com/api",
  Disney = "https://disney.example.com/api"
}
// movies.api.ts
import axios from "axios";

import { APIService } from "@movieApp/api/services";
import { Movie } from "@movieApp/entities";

type MovieDetailsFilters = {/* data model */};

type MovieDetailsRequest = {/* data model */};
type MovieDetailsResponse = {/* data model */};

export async function getMovieDetails(
  movieId: Pick<Movie, "id">,
  filters?: MovieDetailsFilters
): Promise<Movie> {
  const omdbFilters: MovieDetailsRequest = filters
    .map(/* map filters data to omdb model */);

  return axios
    .get(`${APIService.OMDb}/movies/${movieId}`, omdbFilters)
    .then((data: MovieDetailsResponse) => {
      /* map data here to movie model */
    });
}
// someplace.ts
import * as MoviesAPI from "@movieApp/api/movies";

MoviesAPI.getMovieDetails(/* movieId */)

// Demo -- Queries

// Demo -- Queries

// movies.queries.ts
import Store from "@moviesApp/queries/store";

import * as MoviesAPI from "@movieApp/api/movies";
import * as UsersAPI from "@movieApp/api/users";

import { Movie, StreamPlatform } from "@movieApp/entities";

type MovieSummary = Pick<Movie, "id" | "title" | "posterUrl" | "plot">;
type MovieDetails = Movie;

type MovieId = Pick<Movie, "id">;

// Demo -- Queries

// movies.queries.ts
export async function getSummary(movieId: MovieId): Promise<MovieSummary> {
  if (!Store.movies[movieId]) {
    const movieData = await MoviesAPI.getSummary(movieId);

    Store.movies[movieId] = movieData;
  }

  return Store.movies[movieId];
}

export async function getDetails(movieId: MovieId): Promise<MovieDetails> {
  return MoviesAPI.getDetails(movieId);
}

export async function getNetflixStream(movieId: MovieId): Promise<string> {
  return MoviesAPI.getStream(movieId, StreamPlatform.Netflix);
}

// Demo -- Queries

// movies.queries.ts
export async function setLike(movieId: MovieId): Promise<boolean> {
  return UsersAPI.likeMovie(movieId)
    .then(() => {
      Store.user["likes"].push(movieId);
    })
    .then(() => true)
    .catch(() => false);
}

export async function setRating(movieId: MovieId): Promise<boolean> {
  return UsersAPI.setRating(movieId)
    .then((data) => {
      Store.user["rantings"][movieId] = data;
    })
    .then(() => true)
    .catch(() => false);
}

// Demo -- Big Picture

// Conclusions -- PROS

When dealing with codebase changes, it will reduce complexity and make it easier to test and catch bugs early.

It makes it easier to transition between libraries, frameworks or sharing between teams.

It allows to adapt with observables easily and move from a passive programming style to a more reactive one

// Conclusions -- CONS

Codebase ends up being larger and having more dependencies (each layer becomes a dependency)

Requires more engineering discipline and guidelines to keep consistency

Raises cognitive complexity

// Conclusions

Always take everything with a grain of salt, there is not silver bullet or magical wand that is going to make things perfect. Change is constant.

Thank you!

Data Driven Frontend

By Alex Albu

Data Driven Frontend

  • 216