Как я делал сайт визитку на Angular

10e6e28d5e22c812101e72da8b2f1c74.jpg

Несколько месяцев назад я загорелся желанием написать небольшой pet-проект, который был бы посвящен разработке сайта визитки на Angular. И так как Angular достаточно громоздкий фреймворк, в котором нет SSR* из коробки, да и настройка SEO требует немалых телодвижений**, то сама идея выглядела достаточно сомнительной.

Как говорил современник — Выбор — делать дичь, прям лютую грязь-грязь, либо делать грязь, но не прям совсем грязь. 

Сегодня я расскажу, что из этого вышло и стоит ли делать сайты визитки на Angular.

* Universal не берем в расчет, так как он накладывает ряд ограничений при разработке, которые необходимо соблюдать, без этого бесшовная интеграция невозможна.
** Вам придется написать свой генератор sitemap, или по крайней мере составить ее руками, добавить robots, manifest, meta теги и прочее.

В результате получилось небольшое приложение, которое посвящено продаже кроссовок.

72f46bfe8b29b3da19a00f0ffb4950bc.gif

Из-за того, что существуют такие инструменты как Adobe Phpotoshop Gimp и Adobe Premiere Davinci Resolve, выше представленное может быть фейком и поэтому я решил развернуть проект на сервере, где можно потыкать приложение.

С демо проекта можно ознакомиться здесь — https://banshop.fafn.ru.

Исходники на github: https://github.com/Fafnur/banshop

В качестве сервера используется ubuntu, на которой установлен nginx и nodejs. В качестве процесс менеджера использовался pm2.

Арпиори настраивать сервер руками является плохой практикой. Сейчас есть такие инструменты как kubernates, которые могут взять на себя всю рутину и дать полноценный масштабируемый кластер. 

Однако, я подумал, что если мне нужен кластер для разворачивания сайта визитки, то точно моя электричка везёт меня туда, куда я не хочу.

Формирование требований к приложению

Сайт визитка — это расплывчатое понятие. В данном случае под сайтом визиткой будем понимать небольшое web приложение, которое выводит информацию о «компании», а также содержит список товаров, которые можно купить, оформив заказ на сайте. 

Делать полноценный e-commerce я не решился за бесплатно, и поэтому немного ограничил требования. 

Приложение должно содержать следующие модули:

  1. Модуль товаров — каталог товаров, которые выводятся в виде списка карточек товаров. Карточка товара ведет на страницу с детальным описанием товара.

  2. Модуль корзины — функциональность приложения, которая позволяет добавлять в корзину товары с выбранными опциями (размеры, цвета, …). Страница корзины должна отображать список товаров добавленных в корзину, а также предоставлять функциональность изменения/удаления товаров из корзины.

  3. Модуль оформления заказа — функциональность приложения, которая позволяет пользователю указать персональную информацию о себе и оформить заказ, где в заказе будут товары, которые были добавлены в корзину ранее.

  4. Модуль службы поддержки — отдельная страница приложения, где пользователь может связаться с оператором и задать интересующие его вопросы.

  5. Модуль правовой информации — страница, на которой будет представлена информация о продавце (компании), а также порядок продажи товаров и принципы использования информационного ресурса.

С точки зрения отображения, приложение должно быть реализовано для трех платформ:

Так как приложение применяется для онлайн продаж, то необходимо реализовать SEO настройку приложения, которое должно помочь в продвижении сайта в поисковых системах. Список требований к SEO:

  • Приложение содержит карту сайта (sitemap), в которой указаны основные страницы приложения, а также включены все страницы с реализуемыми товарами.

  • Добавлен robots.txt для настройки правил индексации поисковыми роботами

  • Добавлен файл manifest, который поможет настроить PWA в дальнейшем.

  • Для каждой уникальной страницы должны быть добавлены мета теги (keywords, title, description) включая мета теги для og.

  • Поисковая система должна получать готовую, отрисованную страницу (SSR), а не пустой HTML файл с подключенным файлом javascript.

Конечно, еще хотелось добавить требования вида фильтрации товаров по выбранным фильтрам, но видимо это будет только в платной версии.

Выбор тематики, продукции и дизайна

Сначала я хотел сделать сайт визитку на Angular с яхтами и шлюпками, но потом понял, что я далек от яхт. И поэтому в качестве тематики я выбрал продажу кроссовок. 

Дальше дело стояло за дизайном приложения. Так как дизайнер из меня посредственный, я изучил сайты и приложения популярных спортивных брендов и прикупил себе несколько пар кроссовок. Вспомнив о первоначальной задаче, я остановился на бренде Reebok.

Не все же Дудю бесконечно пиарить Adidas.

Если вы разрабатывали сайты на заказ, то знаете, что компетенции людей использующие их в дальнейшем — очень ограничены. Поэтому, для того, чтобы упростить управление сайтом, было принято решение разместить каталог товаров в гугл таблице и дать к ней доступ владельцу ресурса, чтобы он сам мог менять товары, не трогая идеально разработанный код.

Взяв несколько товаров с сайта Reebok и бесчестно украдя один из логотипов компании, я составил табличку в Google Sheet. 

Хотя бренды тратят много денег на маркетинг, но как ни крути дизайн у них в большинстве случаев полный отстой. Мне хотелось чего-то легкого и простого, что было бы понятно программисту, но и от которого не воротило глаз.

И тогда я обратился к лучшему другу всех программистов и посмотрел рекомендации Google. Material Design выглядел ничего, поэтому я взял Angular Material за основу UI KIT«а и решил добавить пару кастомных компонентов. 

Старт проекта

Когда выбор предметной области остался позади, а требования стали конкретны и понятны, то тогда стало возможно перейти к самому интересному — реализации проекта.

Я создал новый репозиторий на гитхабе — banshop. Первый коммит был сделан 24 января 2022 года. Прошло более двух месяцев и я закончил проект, где на момент написания статьи 176 коммитов. 

Думаю, если я бы работал в аутсорсинговой компании, то меня непременно бы уволили за неправильную оценку времени, так как я планировал сделать проект за неделю. На разработку ушло около двух месяцев, где вечерами я шел с работы на работу делать свое небольшое приложение.

Стек технологий, который был использован в приложении:

  1. Nx — инструменты для создания и управления монорепозиторием.

  2. Angular — фронтенд фреймворк.

  3. Ngrx — одна из реализаций Redux в Angular.

  4. Universal — реализация Server Side Rendering.

  5. Angular Localization — модуль локализации.

  6. Jest — фреймворк для unit тестирования.

  7. Cypress — фреймворк для e2e тестирования.

Внешние библиотеки, помимо описанных выше, но без которых сложно обойтись:

  1. Hammerjs — библиотека для отслеживание тапов, свайпов и т.д.;

  2. @angular-builders/custom-webpack — кастомизация билдеров Angular«а;

  3. ng-mocks — библиотека для мокирования сущностей в Angular;

  4. ts-mockito — библиотека для мокирования чего угодно в Typescript;

  5. angular-imask — библиотека с масками.

Все остальные пакеты в приложении — это вкусовщина (husky, eslint,…).

Если почитать другие мои статьи, то весь выше описанный стек есть почти в каждом моем проекте. Одним словом у меня есть определенный стек и я его придерживаюсь.

Процесс разработки приложения

Весь процесс разработки можно разбить на несколько шагов:

  1. Создание workspace с установкой всех необходимых библиотек.

  2. Создание core модулей.  Так как каждый разработчик придерживается определенных решений, то на старте проекта есть смысл заложить библиотеки и решения как часть ядра разрабатываемого приложения. 

  3. Создание UI KIT, который должен содержать все общие компоненты, которые планируется использовать в приложении. Ярким примером являются модули сеток, контейнеров, слайдеры, ссылки, кнопки и прочее.

  4. Реализация функциональных модулей, таких как модуль товаров, модуль корзины, модуль чата, модуль оформления заказа.

  5. Настройка SEO, где в приложение добавляются все необходимые файлы и настройки для поисковой оптимизации.

  6. Настройка SSR и пререндера. На данном шаге производится заточка всего приложения под корректную работу SSR.

  7. Тестирование. Конечно, тесты пишутся вместе с реализацией модулей, но есть некоторые места c TODO: Add test, которые на данном шаге нужно устранить. Да и e2e разрабатываются на этом шаге, так как их написание на более ранних шагах бессмысленна.

  8. Оптимизация приложения с точки зрения используемых ресурсов и поддержки современных веб стандартов. Как одним из инструментов можно использовать Lighthouse от Google chrome.

Вкратце опишу каждый из шагов, чтобы было понимание того, что пришлось сделать, чтобы сайт заработал.

Отмечу, что весь процесс разработки подробно описан в моем цикле статей на медиуме.

Создание workspace

Создания нового workspace очень легко, где нужно запустить одну команду и NX CLI создадут новый workspace и если выбрать опцию с Angular, то еще будет сгенерировано новое Angular приложение.

Достаточно запустить следующую команду:

yarn create nx-workspace --package-manager=yarn

Если добавить библиотеки вышеописанного стека с помощью nx, то тогда в приложении будут автоматически созданы необходимые файлы конфигурации.

Например конфигурация SSR:

yarn nx add @nguniversal/express-engine

Для настройки Ngrx достаточно создать Root State, которая добавит в проект необходимые зависимости.

nx g @nrwl/angular:ngrx rs --module=path/to/app.module.ts

Настройка локализации:

nx add @angular/localize

Настройка Angular Material:

nx add @angular/material

Установка внешних библиотек:

yarn add hammerjs angular-imask

Установка дополнительных библиотек:

yarn add -D ng-mocks ts-mockito

И немного вкусовщины:

yarn add -D eslint-plugin-import eslint-plugin-jsdoc eslint-plugin-ngrx eslint-plugin-prettier eslint-plugin-simple-import-sort

Запустив все команды и применив немного магии для eslint«а, то тогда создаться новое рабочее пространство (workspace) с Angular приложением.

Создание core модулей

Core модули — набор решений, которые позволяют решить узкие места фреймворка. 

Примеры узких мест:

  • Получение доступа к Window;

  • Реализация кроссплатформенных хранилищ localStroge, sessionStrorage;

  • Интерфейсы и утилиты для создания тестовых объектов (PageObject) и DI мок-сервисов;

  • Унификация навигации в приложении;

  • Маппинг FormControl из FormGroup;

  • ApiService — обертка над HttpClient, которая управляет созданием корректных путей и логированием ошибок при необходимости.

Большая часть core — это системные части, которые расширяют базовые возможности фреймворка или решают проблемы с платформами, такими как предоставление доступа к web хранилищам в серверной платформе, в которой нет window.

В качестве примера приведу решение с доступом к window:

import { DOCUMENT } from '@angular/common';
import { Inject, Injectable } from '@angular/core';

@Injectable({
  providedIn: 'root',
})
export class WindowService {
  constructor(@Inject(DOCUMENT) public readonly document: Document) {}

  get window(): Window | null {
    return this.document.defaultView;
  }
}

Теперь вне зависимости от платформы, можно обращаться к window:

const navigator = this.windowService.window?.navigator

Конечно, в серверной платформе window будет null, но лучше null, чем ошибка компиляции приложения.

В сервис можно добавить проверку на доступность window и привести сервис только к явному использованию браузерной версии:

import { DOCUMENT } from '@angular/common';
import { Inject, Injectable } from '@angular/core';

@Injectable({
  providedIn: 'root',
})
export class WindowService {
  constructor(@Inject(DOCUMENT) public readonly document: Document) {}

  get window(): Window {
    const window: Window | null = this.document.defaultView;

    if (window === null) {
      throw new Error('Default view is not defined!');
    }

    return window;
  }
}

Остальные решения, которые представлены в core модулях также тривиальны, как и решение с Window. Ознакомиться с ними можно на гитхабе — banshop/core.

Разработка UI KIT

Возможно одним из главных решений при разработке приложения, которое будет влиять на стоимость его поддержки — это разработка UI KIT.

UI KIT — набор общих компонентов, которые используются для построения приложения.  

Если сначала не определить общие компоненты, то в дальнейшем, похожие компоненты будут дублировать код, для реализации похожего функционала. И чем больше будет становиться приложение, тем больше придется заниматься поддержкой проекта. 

А так как мой внутренний перфекционист требует масштабирования бизнеса, то UI KIT для сайта визитки жизненно необходим.

Так как я решил использовать в качестве основы Angular Material, то уже из коробки достаточно много доступных компонентов.

Сформировать тему для material очень легко. Это делается в момент установки пакета материала.

Единственное, что на мой взгляд плохо освещено в документации по Angular Material — это создание лейаутов. Подробнее о лейаутах в документации.

Angular Material предоставляет возможность создать три платформы для отображения с помощью средств CDK. В CDK есть константа Breakpoints со следующими размерами:

export declare const Breakpoints: {
    XSmall: string;
    Small: string;
    Medium: string;
    Large: string;
    XLarge: string;
    Handset: string;
    Tablet: string;
    Web: string;
    HandsetPortrait: string;
    TabletPortrait: string;
    WebPortrait: string;
    HandsetLandscape: string;
    TabletLandscape: string;
    WebLandscape: string;
};

Если сделать ряд манипуляций и создать сервис, который используя BreakpointObserver будет возвращать текущий тип платформы.

Сервис может быть реализован следующим образом:

import { BreakpointObserver, Breakpoints } from '@angular/cdk/layout';
import { Injectable } from '@angular/core';
import { BehaviorSubject, Observable, tap } from 'rxjs';

export const LAYOUT_SHORT_TYPES_MAP = {
  [Breakpoints.Handset]: Breakpoints.Handset,
  [Breakpoints.HandsetPortrait]: Breakpoints.Handset,
  [Breakpoints.HandsetLandscape]: Breakpoints.Handset,
  [Breakpoints.Tablet]: Breakpoints.Tablet,
  [Breakpoints.TabletPortrait]: Breakpoints.Tablet,
  [Breakpoints.TabletLandscape]: Breakpoints.Tablet,
  [Breakpoints.Web]: Breakpoints.Web,
  [Breakpoints.WebPortrait]: Breakpoints.Web,
  [Breakpoints.WebLandscape]: Breakpoints.Web,
};

export const LAYOUT_TYPES = [Breakpoints.Handset, Breakpoints.Tablet, Breakpoints.Web];

@Injectable({
  providedIn: 'root',
})
export class LayoutService {
  private readonly layoutSubject$ = new BehaviorSubject(Breakpoints.Handset);

  get layoutType$(): Observable {
    return this.layoutSubject$.asObservable();
  }

  get snapshotLayoutType(): string {
    return this.layoutSubject$.value;
  }

  constructor(private readonly breakpointObserver: BreakpointObserver) {
    this.breakpointObserver
      .observe(LAYOUT_TYPES)
      .pipe(
        tap((result) => {
          let type;
          for (const query of Object.keys(result.breakpoints)) {
            if (result.breakpoints[query]) {
              type = LAYOUT_SHORT_TYPES_MAP[query];
              break;
            }
          }

          this.layoutSubject$.next(type ?? Breakpoints.Handset);
        })
      )
      .subscribe();
  }

  is(size: string): boolean {
    return size === this.snapshotLayoutType;
  }
}

Вышеописанные шаги позволят реализовать решение, которое позволит использовать три состояния, для отображения контента, в частности мобильного телефона, планшета и ПК.

77b9048e84c39f7e151fcf4eb4930d59.jpeg

Помимо лейаута я создал несколько компонентов для создания выравнивания контента на странице: GridModule и ContainerModule.

Первый модуль позволяет создавать сетки по аналогии с сетками в Twitter Bootstrap.

Второй модуль реализует аналог класса контейнера из Twitter Bootstrap, где у контента есть фиксированная ширина и блок позиционируется по центру.

И последними компонентами в UI KIT стали модули для отображения карусели. Карусель реализована топорно, где выводится список полученных изображений, и добавлены элементы навигации.

ebe11ba648cb03c31f3691cffa65c846.gif

Реализация функциональных модулей

Реализация всех функциональных модулей, таких как модуль товаров, модуль корзины, модуль оформления заказа и модуль чата.

05c87d8232550c981714b1204cac0eef.gif

Я использовал следующую последовательность действий для разработки каждого модуля.

Сначала создавалась библиотека, которая включала в себя все интерфейсы и абстракции для текущего модуля. 

Вынесение всех сущностей в отдельную библиотеку имеет одно весомое преимущество — это позволяет разделить абстракции и реализации друг от друга, при этом шансы создания циклических зависимостей минимальны.

После того, как были созданы все интерфейсы, создавался новый state (feature state), который включал в себя всю логику с изменением данных для модуля. В процессе разработки state были созданы экшены и сформирован редьюсер, описаны все состояния state, а также были реализованы цепочки событий с помощью эффектов и добавлены соответствующие методы в фасад.

Следующим шагом было создание UI компонентов для каждого из модулей. Сначала создавались видимые модули, которые использовались на страницах модулей — страница товаров, страница корзины, страница оформления заказов, а уже потом все остальные компоненты, которые добавляли вспомогательный функционал. Примеры вспомогательного функционала — это кнопки добавления в корзину, где при клике добавить товар в корзину, открывалось диалоговое окно, в котором отображалась карточка товара и была возможность добавить товар в корзину с выбором вариационных полей.

После формирования UI компонентов создавалась страница модуля, где подключались ранее созданные компоненты из UI KIT и UI компонентов модуля.

В конце разработки модуля создавались гуарды (can-active), которые проверяли права доступа к страницам.

Настройка локализации

Я же в проекте, добавил полноценную локализацию, где в качестве базового языка используется рунглиш английский, а само приложение использует русский язык. 

Это единственный шаг, который можно было не делать совсем — это прикручивать локализацию в Angular. Точнее сказать, сконфигурировать базовые настройки локализации нужно, но делать шаблоны мультиязычными — точно нет.

Так как приложение не подразумевает другие языки, то потребность в переводе отсутствует. Я добавил локализацию только потому, что не хотел использовать русский в шаблонах и компонентах.  С другой стороны получилось показать на практике как использовать стандартную локализацию, а также привести пример того, какие части приложения не стоит локализовать.

Локализация Angular работает с помощью использования файла локализации, который генерируется на основе поиска локализованных частей приложения с помощью применения пайпа i18n и глобальной переменной $localize.

Для того, чтобы добавить локализацию в Angular нужно всего-лишь добавить пакет:

nx add @angular/localize

И в браузерной версии, все будет замечательно. Помечаете части приложения с локализацией специальным пайпом (i18n) и генерируете файл локализации.

Пример файла локализации:



  
    
      
        Cart | Online store Banshop
        Корзина | Online store Banshop
        
          libs/cart/page/src/lib/cart-page-routing.module.ts
          17
        
        Cart meta
      
    











  

Проблемы начинаются, когда появляется SSR. Не знаю с чем это связано, но сейчас не найти примера приложения angular + universal + localization в официальной документации. 

Если погуглить локализацию, то можно найти мои статьи 2019 года, в которых я привожу примеры настройки связки angular + universal + localization.

Не могу не отметить, что единственная полезная статья у меня в блоге на медиуме. Я занимаюсь настройкой локализации примерно раз в год, и спустя год уже забываешь все конфигурации и нюансы. И статья меня уже дважды выручала.

Если в двух словах, то при добавлении локализации в univesral появляются префиксы при билде es/ru/en-IN. И если серверная сборка адекватно может обрабатывать локализацию, то dev-server не может. И тут начинают танцы с бубном, как заставить дев сервер работать. 

Решением является выключение локализации для серверной версии, где в конфигурациях серверной версии уже руками для каждой конфигурации прописывается локаль, и создается одна конфигурация без локали.

Настройка SEO

Как я и писал выше, для Angular необходимо выполнить несколько шагов, для того, чтобы поисковые системы стали адекватно индексировать приложение.

Сначала я создал набор favicon«ов, где вместе с набором иконок создались файлы манифеста.  Далее создал robots.txt, в котором запретил индексировать пути к API.

Следующей задачей стала реализация мета тегов.

Так как Angular не предлагает готового решения, пришлось изобрести свой велосипед и уехать в глухомань.

Если грубо очертить, то нужен сервис, который в зависимости от страницы приложения указывал следующий набор мета тегов и сопутствующих тегов: canonical, title, description, keywords. Мета теги OG: title, description, type, locale, siteName, image, imageType, imageWidth, imageHeight.

Реализация подобного сервиса может быть следующей:

import { DOCUMENT } from '@angular/common';
import { Inject, Injectable, LOCALE_ID, Optional } from '@angular/core';
import { Meta, MetaDefinition, Title } from '@angular/platform-browser';
import { Router } from '@angular/router';

import { EnvironmentService } from '@banshop/core/environments/service';
import {
  META_CONFIG,
  META_CONFIG_DEFAULT,
  META_CONFIG_OG,
  META_CONFIG_OG_DEFAULT,
  MetaConfig,
  MetaConfigOg,
} from '@banshop/core/meta/common';

@Injectable({
  providedIn: 'root',
})
export class MetaService {
  private readonly metaConfig: MetaConfig;
  private readonly metaConfigOg: MetaConfigOg;

  constructor(
    private readonly titleService: Title,
    private readonly router: Router,
    private readonly meta: Meta,
    private readonly environmentService: EnvironmentService,
    @Inject(DOCUMENT) private readonly document: Document,
    @Inject(LOCALE_ID) private readonly localeId: string,
    @Optional() @Inject(META_CONFIG) metaConfig: MetaConfig | null,
    @Optional() @Inject(META_CONFIG_OG) metaConfigOg: MetaConfigOg | null
  ) {
    this.metaConfig = metaConfig ?? META_CONFIG_DEFAULT;
    this.metaConfigOg = metaConfigOg ?? META_CONFIG_OG_DEFAULT;
  }

  update(metaConfig?: Partial, metaConfigOg?: Partial): void {
    const config: MetaConfig = { ...this.metaConfig, ...metaConfig };
    const configOg: MetaConfigOg = { ...this.metaConfigOg, ...metaConfigOg };
    this.setCanonicalUrl(config.url);
    this.titleService.setTitle(`${config.title} | ${this.environmentService.environments.brand}`);
    this.setMetaProperty('description', config.description);
    this.setMetaProperty('keywords', config.keywords);
    this.setMetaProperty('og:title', `${configOg.title ?? config.title} | ${this.environmentService.environments.brand}`);
    this.setMetaProperty('og:description', configOg.description ?? config.description);
    this.setMetaProperty('og:type', configOg.type);
    this.setMetaProperty('og:locale', configOg.locale ?? this.localeId);
    this.setMetaProperty('og:site_name', configOg.siteName ?? this.environmentService.environments.brand);
    this.setMetaProperty('og:image', `${this.environmentService.environments.appHost}${configOg.image}`);
    this.setMetaProperty('og:image:type', configOg.imageType);
    this.setMetaProperty('og:image:width', configOg.imageWidth);
    this.setMetaProperty('og:image:height', configOg.imageHeight);
  }

  private setCanonicalUrl(url?: string): void {
    const link = (this.document.getElementById('canonical') ?? this.document.createElement('link')) as HTMLLinkElement;
    link.setAttribute('rel', 'canonical');
    link.setAttribute('id', 'canonical');
    link.setAttribute('href', this.getCanonicalURL(url));
    if (!this.document.getElementById('canonical')) {
      this.document.head.appendChild(link);
    }
  }

  private getCanonicalURL(url?: string): string {
    return `${this.environmentService.environments.appHost}${url ?? this.router.url}`;
  }

  private setMetaProperty(name: string, content: string): void {
    const id = `meta-${name}`;
    const has = !!this.document.getElementById(id);

    const meta: MetaDefinition = { id, name, content };

    if (has) {
      this.meta.updateTag(meta);
    } else {
      this.meta.addTag(meta);
    }
  }
}

Я пробовал разные способы задания мета тегов, но лучшим решением является создание мета тегов при конфигурации роута страницы.

import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router';

import { RouteData } from '@banshop/core/navigation/common';

import { CartPageComponent } from './cart-page.component';

const routes: Routes = [
  {
    path: '',
    component: CartPageComponent,
    data: {
      sitemap: {
        loc: '/cart',
        priority: '1.0',
      },
      meta: {
        title: $localize`:Cart meta|:Cart | Online store Banshop`,
        description: $localize`:Cart meta|:It is very easy to buy on banshop. To place an order, click the order button.`,
        keywords: $localize`:Cart meta|:cart, banshop`,
      },
    } as Partial,
  },
];

@NgModule({
  imports: [RouterModule.forChild(routes)],
  exports: [RouterModule],
})
export class CartPageRoutingModule {}

Для того, чтобы теги изменялись сами, я обычно создаю новый класс ngrx эффектов, которые подписываются на окончание смены роута и после этого, вызывается соответствующий метод сервиса, который обновляет мета теги для страницы.

Последней задачей с SEO — это создание карты сайта. 

Снова Angular предоставляет вам возможность блеснуть талантом и сделать все самому. 

Однажды я написал решение на NodeJS, которая просматривает все файлы в проекте и ищет модули, которые настраивают навигацию в приложении. В найденных файлах ищется конфиг в data, который описывает требования к генерации пути в sitemap. Если конфигурация найдена, то данный путь добавляется в карту сайта.

Пример данного решения:

import { config } from 'dotenv';
import * as fs from 'fs';
import * as https from 'https';
import * as path from 'path';

import { SitemapConfig } from '@banshop/core/navigation/common';

import { environment } from './src/environments/environment.prod';

config({ path: 'apps/store/.env' });

const routes = new Set();

/**
 * Find file on folders
 */
export function fromDir(startPath: string, filter: string): string[] {
  if (!fs.existsSync(startPath)) {
    console.warn('no dir ', startPath);
    return [];
  }
  const founded = [];
  const files = fs.readdirSync(startPath);
  for (const file of files) {
    const filename = path.join(startPath, file);
    const stat = fs.lstatSync(filename);
    if (stat.isDirectory()) {
      const foundedIn = fromDir(filename, filter);
      founded.push(...foundedIn);
    } else if (filename.indexOf(filter) >= 0) {
      founded.push(filename);
    }
  }

  return founded;
}

/**
 * Find sitemap config on file content
 */
export function parseSitemapConfig(source: string): Partial {
  let sitemapConfig = source
    .slice(8)
    .replace(/\n|\t|\s/g, '')
    .replace(/'/g, '"')
    .trim();
  if (sitemapConfig[sitemapConfig.length - 1] === ',') {
    sitemapConfig = sitemapConfig.slice(0, sitemapConfig.length - 1);
  }
  sitemapConfig = sitemapConfig + '}';
  sitemapConfig = sitemapConfig.replace(
    /(\w+:)|(\w+ :)/g,
    (matchedStr: string) => '"' + matchedStr.substring(0, matchedStr.length - 1) + '":'
  );

  return JSON.parse(sitemapConfig);
}

/**
 * Generate sitemap url
 */
export function getSitemapUrl(sitemap: Partial): string {
  if (sitemap.loc) {
    routes.add(sitemap.loc.length > 0 ? sitemap.loc : '/');
  }
  return `${environment.appHost}${sitemap.loc}${(sitemap.lastmod
    ? new Date(sitemap.lastmod)
    : new Date()
  ).toISOString()}${sitemap.changefreq ?? 'daily'}${sitemap.priority ?? 0.8}`;
}

/**
 * Load data and generate sitemap config
 */
export function getServerData(cb: (data: string) => void): void {
  let data = '';
  const { GOOGLE_KEY, GOOGLE_ID, GOOGLE_NAME } = process.env;

  if (GOOGLE_KEY && GOOGLE_ID && GOOGLE_NAME) {
    const options = {
      hostname: 'sheets.googleapis.com',
      path: `/v4/spreadsheets/${GOOGLE_ID}/values/${GOOGLE_NAME}?key=${GOOGLE_KEY}`,
      method: 'GET',
    };

    const req = https.request(options, (res) => {
      let body = '';

      res.on('data', (chunk) => {
        body += chunk;
      });

      res.on('end', () => {
        const response = JSON.parse(body);

        if (response.values) {
          for (const product of response.values) {
            data += getSitemapUrl({
              loc: `/product/${product[0]}`,
              lastmod: new Date().toISOString(),
              changefreq: 'daily',
              priority: '0.8',
            });
          }
        }
        cb(data);
      });
    });

    req.end();
  }
}

export function getUrls(): string {
  let data = '';
  const files = [...fromDir('./apps/store/src', '-routing.module.ts'), ...fromDir('./libs', '-routing.module.ts')];

  for (const file of files) {
    const fileContent = fs.readFileSync(file, 'utf8');
    const sources = fileContent.replace(/\s+/, ' ').match(/sitemap:\s{[^}]+/g);

    if (sources) {
      for (const source of sources) {
        data += getSitemapUrl(parseSitemapConfig(source));
      }
    }
  }

  return data;
}

export function generate(): void {
  const urls = getUrls();

  getServerData((data) => {
    fs.writeFileSync(
      'apps/store/src/sitemap.xml',
      // eslint-disable-next-line max-len
      `${urls}${data}`
    );
    const routePaths = [...Array.from(routes), '/not-found', '/server-error'].sort().join('\n');
    fs.writeFileSync('apps/store/routes.txt', routePaths);
  });
}

// generate
generate();

Генератор содержит несколько функций:

  • fromDir — читает содержимое директории и ищет файлы по шаблону;

  • parseSitemapConfig — парсит содержимое файла и пытается найти конфигурацию sitemap;

  • getSitemapUrl — генерация пути в sitemap на основании входящего конфига;

  • getServerData — загрузка внешнего файла и генерация url«ов на основе полученных данных;

  • getUrls — функция, которая парсит приложение и библиотеки для конкретного приложения на наличие файлов роутинга;

  • generate — функция, которая запускает генерацию карты сайта.

Так как пример написан на Typescript, то необходимо добавить tsconfig, чтобы скрипт можно было запустить из корня проекта.

{
  "extends": "../../tsconfig.base.json",
  "files": [],
  "include": [],
  "compilerOptions": {
    "module": "CommonJS",
    "lib": ["DOM", "ESNext"]
  }
}

Пример генерации карты сайта:



  
    https://banshop.fafn.ru/cart
    2022-03-19T09:38:02.917Z
    daily
    1.0
  

Помимо статичных путей, в карту сайта попали и динамические пути. Это решение применимо только для этого решения, так как список всех товаров находится в Google Sheets, то во время выполнения скрипта можно запустить загрузку файла и потом в добавить в карту сайта все динамические страницы. 

Из важного, стоит отметить, что вместе с генерацией карты сайта, генерируется массив страниц для prerender«а:

const routePaths = [...Array.from(routes), '/not-found', '/server-error'].sort().join('\n');
fs.writeFileSync('apps/store/routes.txt', routePaths);

Это позволяет в фазе prerender«а отрисовать все требуемые страницы в приложении.

Настройка SSR

580f4586a86cba2de322f563c697c8f6.png

Одним из мотиваторов разработки приложения было именно показать, как настраивать и использовать Universal.

Настройка SSR — один из моих любимых вопросов на собеседовании. Правда со мной на эту тему никто не хочет разговаривать.  

Как было сказано ранее, большой плюс Angular это схематики и настройки. При добавлении SSR, большая часть настроек будет сконфигурирована.

yarn nx add @nguniversal/express-engine

Однако, если не соблюдать правила разработки под SSR, то приложение не запуститься.

Какие нюансы стоит учитывать, если планируется использовать SSR:

  • Доступ к Window. Везде, где используются web технологии (localstrorage, navigator, …) необходимо ставить проверки на платформу, что код будет запускаться только в браузерной версии приложения.

  • Контроль observable, которые не должны уходить в бесконечный цикл. Например использование interval без обертки платформы или правил, которые принудительно завершат поток приведет к тому, что приложение не будет собрано, так как процесс компиляции (чаще пререндер) повиснет на стадии ожидания завершения потока, который никогда не завершится.

  • Контроль observable guard«ов, которые ожидают данные, но которые могут быть не получены. В данном случае, тоже умрет пререндер, который не сможет дождаться завершения потока.

  • При разработке стилей, все глобальные стили в приложении будут грузится с помощью style.css, а это значит, что элементы на странице могут дергаться после загрузки страницы. Поэтому есть смысл сократить использование глобальных стилей, оставив там только ключевые стили для Angular CDK и Angular Material

  • Контроль state приложения.

Так как в Universal есть пререндер, то нужно сделать еще ряд манипуляций, чтобы заставить приложение на NodeJS раздавать отрендеренные страницы вместо полноценной отрисовки.

90a6f6b6c1549878f31b371c52e6687b.png

Теперь все по шагам, что было сделано в приложении.

После установки зависимостей я немного переименовал файлы и создал еще один файл модуля для приложения:

  • app.module.ts — общий модуль приложения, который используется как в браузерной версии, так и в серверной

  • app.browser.module.ts — модуль только для браузреной версии. Модуль может подключать библиотеки, которые работают только в браузере, но не будут корректно работать в среде node

  • app.server.module.ts — модуль только для сервреной версии. В данной платформе можно использовать path, fs и все остальные плюшки и библиотеки от nodejs.

Переименовал main.ts  в main.browser.ts, чтобы было понимание, какая платформа запускается.

В самом файле сервера, я изменил правила отрисовки страниц:

// All regular routes use the Universal engine
  server.get('*', (req, res) => {
    const filePath = join(distFolder, req.path, 'index.html');

    // For prerender, use exists file
    if (existsSync(filePath)) {
      res.sendFile(filePath);
    } else {
      res.render(indexHtml, {
        req,
        providers: [
          {
            provide: APP_BASE_HREF,
            useValue: req.baseUrl,
          },
          {
            provide: REQUEST,
            useValue: req,
          },
          {
            provide: RESPONSE,
            useValue: res,
          },
        ],
      });
    }
  });

В данном случае, сначала по заданному пути ищется соответствующий файл, который мог быть создан prerender«ом. Если такой файл есть, то тогда он отдается сервером, иначе запускается полноценная отрисовка приложения на сервере.

Запуск отрисовки стрницы сервером — очень дорогая операция.

Только если вы не Джеф Безос и можете позволить себе сервера на амазоне.

Если тестировать локально, то время отдачи главной страницы SSR — 800 милисекунд.

3027dcd7442395753847fa35c668467c.png

Если же отдавать страницу, созданную пререндером, то тогда ответ сервера 20 миллисекунд.

26c44a43af0d0222335e4d96e1a3a32b.png

Разница ощутима.

А если еще делегировать отдачу статики на балансировщик, где статичные файлы будут отдаваться напрямую nginx, то можно ускорить от сервера еще в несколько раз. И тогда не будет необходимости проверять файлы статики с помощью ноды. 

Последний нюанс связанный с SSR — это передача состояния с серверной платформы в браузерную версию.

Например, в данном приложении запрашивается список товаров, который потом складывается в Ngrx state. Если отрисовать страницу пререндером, который успешно загрузит данные по API, то в ответе от сервера, клиенту прилетит готовая страница.

Однако, так как в браузерной версии нет данных от API, то при запуске страницы, будет выполнен запрос на загрузку данных по API еще раз, но в этот раз с браузерной версии.

Тут есть несколько решений, чтобы отрисованная страница не пропадала, а потом снова отрисовывалась с теми же данными. 

Вариант 1. Если приложение разработано по уму, и не может быть такого, что браузерное приложение может получить данные, отличные от тех данных, что были получены в серверной версии, то тогда, запрос в браузерной версии можно вообще не делать. Для этого используется TransferState — передаваемые значения между платформами в Angular. Перед выполнением реального запроса, можно проверить значение в TransferState. Если значение в TransferState нету, то тогда необходимо выполнить запрос. 

Для того, чтобы задать значение в TransferState, в серверной платформе, в случае успешного выполнения запроса записывается значение по ключу. И когда, браузерная версия попробует выполнить запрос, она сначала проверит наличие значения, и только если значения не будет, то выполнит запрос.

  load$ = createEffect(() => {
    return this.actions$.pipe(
      ofType(ProductActions.load),
      fetch({
        id: () => 'load-products',
        run: () =>
          this.productApiService.load().pipe(
            map((products) => {
              
    
            

© Habrahabr.ru