За 25 дней нам удалось добиться довольно больших изменений в нашем проекте.

Мы провели:

  • Рефакторинг back-end сервисов

  • Убрали часть легаси кода на фронте

  • Переработали некоторый UI элементы и добавили плавности

  • Добавили новый функционал

Чем бы дитя не тешилось, лишь бы не плакало

Именно так мы подумали и решили завершать первый важный этап нашего MVP проекта. Мы подготовили всю инфраструктуру к работе, подбили UI что бы если и есть баги - то которые мы явно упустили за 30 часов тестирования.

Новый функционал

Мы добавили в приложение следующее:

  1. Рассылка кодов авторизации

  2. Поиск серверных чатов

  3. Подписка у пользователя в сообщениях какая это роль

  4. Переработали дизайн тредов, теперь у него больше настроек

  5. Сохранение сообщений

Поиск общих чатов

Не есть что конечно, слизано с приложения Discord, но мое виденье его внутри компании немного другого формата.

В чем вообще задумка его для корпоративного мессенджера? Для начала легко найти чаты внутри команд или структур. Группировка - это настраиваемые элементы, мы вывели их в отдельный конфиг, так что поправить под компанию - 5 минут. Настраивается из настроек чата.

В будущем добавим и плейсхолдел для настроек, пока что так оставим.
В будущем добавим и плейсхолдел для настроек, пока что так оставим.

Подписка у пользователя в сообщениях какая это роль

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

Переработали дизайн тредов, теперь у него больше настроек

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

  1. Мы расширили настройки на создание тредов

Теперь кроме названия треда и иконки можно:

  • Проставить теги

  • Написать нормальное описание

После создания тредов теперь мы можем не только посмотреть их список, но и так же:

  • Отфильтровать по названию, даже по одному слову в середине текста названия или описания

  • Отфильтровать по тегам, идет сортировка по часто используемым

  • Закрепить тред (все закрепленные всегда будут вверху, только потом будут идти не закрепленные, даже при поступлении в них новых сообщений)

Возможно, предвижу, кому-то удобнее прям внутри сообщений писать, тем самым создавая обсуждения. Потом мы добавим такое, но не в рамки обсуждения, а в рамках внутренних комментариев.

Рассылка кодов авторизации

Инфраструктура под это дело заложена в сервис авторизации. При регистрации или авторизации сервис будет отправлять на почту пользователя сообщение в формате HTML.

Рендер письма который заложен в код процесса рассылки
Рендер письма который заложен в код процесса рассылки

Пока еще не заводили отдельную почту для нашего сервиса.

Рефакторинг back-end сервисов

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

Теперь Gateway (основной сервис который либо пропускает запрос, либо нет) - подключен к redis pub/sub, инвалидирует версию прав и бит маску. Если не было события изменения - берет из мемори, если была - обновляет свой кеш. Кеш одного пользователя занимает 600.B. что не является слишком много. Для тяжелых прав на 2000 пользователей в одном сервер чате это около 4.5мб. В целом - пойдет.

Кеш хранимый в gateway:

{
    'chat_id':  str,          # ObjectId, 24 символа
    'version':  int,          # монотонный счётчик
    'base':     int,          # ← OR всех ролей юзера, ОДНО число
    'topics':   {tid: int},   # только топики с overrides
    'is_owner': bool,         # админ или нет
}
  • Сервис топиков тоже потерпел изменения. теперь он не ходит в Permission сервис что бы уточнить права пользователя на просмотр, а Gateway: ProxyRouter приклеивает заголовки:

def build_perm_headers(self, snapshot: dict) -> dict:
    return {
        'X-Perm-Base':   str(snapshot.get('base', 0)),
        'X-Perm-Topics': json.dumps(snapshot.get('topics', {})),
        'X-Is-Owner':    '1' if snapshot.get('is_owner') else '0',
    }

А сервис топиков их начинает парсить:

def parse_perm_headers(request) -> Optional[dict]:
    base = int(request.headers.get('x-perm-base'))
    topics = json.loads(request.headers.get('x-perm-topics', '{}'))
    is_owner = request.headers.get('x-is-owner') == '1'
    return {'base': base, 'topics': topics, 'is_owner': is_owner}

На выход топик сервис уже отдает отфильтрованные темы которые соответствуют правам пользователя внутри его роли:

{
    "success": true,
    "topics": [
        {
            "topic_id": "69ee3a46dab658745b095c27",
            "chat_id": "69ee32fba41e74b3744d6502",
            "name": "59545",
            "created_by": "69edfdf3a6343fa670fe55db",
            "topic_type": "thread",
            "position": 0,
            "created_at": "2026-04-26T16:16:06.983000",
            "updated_at": "2026-04-28T07:09:30.006000",
            "is_active": true,
            "group_id": "69f05d25335af601c7accfe7",
            "unread_count": 0
        },
        {
            "topic_id": "69ee39a4dab658745b095c26",
            "chat_id": "69ee32fba41e74b3744d6502",
            "name": "55532",
            "created_by": "69edfdf3a6343fa670fe55db",
            "topic_type": "text",
            "position": 1,
            "created_at": "2026-04-26T16:13:24.480000",
            "updated_at": "2026-04-28T07:09:30.001000",
            "is_active": true,
            "group_id": "69f05d25335af601c7accfe7",
            "unread_count": 0
        },
        {
            "topic_id": "69f05cf4335af601c7accfe6",
            "chat_id": "69ee32fba41e74b3744d6502",
            "name": "fgfhd",
            "created_by": "69edfdf3a6343fa670fe55db",
            "topic_type": "voice",
            "position": 2,
            "created_at": "2026-04-28T07:08:36.007000",
            "updated_at": "2026-04-28T07:09:30.001000",
            "is_active": true,
            "unread_count": 0,
            "group_id": null
        }
    ],
    "groups": [
        {
            "chat_id": "69ee32fba41e74b3744d6502",
            "name": "tdrfhg",
            "created_by": "69edfdf3a6343fa670fe55db",
            "position": 0,
            "created_at": "2026-04-28T07:09:25.028000",
            "updated_at": "2026-04-28T07:09:25.028000",
            "is_active": true,
            "group_id": "69f05d25335af601c7accfe7"
        }
    ]
}
  • Наконец то вынесли все коллекции сервис в отдельные базы. Раньше была одна общая БД, делалось для быстрого написания кода, но пораждало много зависимостей и хождений в чужие коллекции. Теперь все сервисы имеют свою БД со своими коллециями. Отдельный инстанс имеет только сервис сообщений.

  • 50% сервисов перевели в режим публикации событий. Т.е. раньше было много HTTP вызовов между ними, что пораждало лишние задержки в 1-4мс. Мелочь, но в рамках 1000 или 10000 пользователей это уже существенная нагрузка. Теперь HTTP вызовы служат только для получения информации от другого сервиса, все события которые раньше были переложены на Redis. Пример: Пользователь зашел в новый чат. Сервис чата сформирует событие что у него появился новый memories, сервис прав подхватит это событие и присвоит ему права и положит их в Redis, сервис Gateway увидит новые данные и заберет их к себе.

    • Мы работаем над переводом остальных сервисов, но не все так быстро))

Удаление легаси кода

При создании проекта, я вообще ничего не знал о методах разработки фронтенда. Сейчас когда нас несколько уже и мы отходим от AI разработки - начинаем сталкиваться с проблемами. Например тогдлы переключения - так как фронт писал ИИ, он в каждую форму где есть тогл - делал новый стилевый файл, и таких файлов скопилось 15+ ед. Мы перевели их на shared и пере-используем. И таких компонентов было очень много. В итоге нам получилось подчистить примерно 2к стилевых строк кода, сократить в JSX около 150 строк, потому что теперь это общие подключения.

Выделили отдельные сервисы кеша. Тонкие обёртки над универсальным key-value store для кэширования. Не хранят данные сами — только формирует ключи и инкапсулирует логику работы с одним конкретным namespace. Пример ниже:

export class TopicCache {
    constructor(store) {
        this.store = store;
        this.KEY = 'topic_messages';
    }

    getMessages(chatId, topicId) {
        return this.store.get(this.store.key(this.KEY, chatId, topicId));
    }

    setMessages(chatId, topicId, messages) {
        this.store.set(this.store.key(this.KEY, chatId, topicId), messages);
    }

    invalidateMessages(chatId, topicId) {
        if (topicId) {
            this.store.invalidate(this.store.key(this.KEY, chatId, topicId));
        } else {
            this.store.invalidateByPrefix(this.store.key(this.KEY, chatId));
        }
    }

    appendMessage(chatId, topicId, message) {
        const cacheKey = this.store.key(this.KEY, chatId, topicId);
        const cached = this.store.get(cacheKey);
        
        if (cached) {
            const exists = cached.some(m => m.message_id === message.message_id);
            if (!exists) {
                const updated = [...cached, message];
                this.store.set(cacheKey, updated);
                console.log(`[CACHE] Appended message to topic ${chatId}:${topicId}, total: ${updated.length}`);
            }
        }
    }
}

Аналогично сделали и с WS сервисом, вынесли все WS действия в отдельные сервисы, для сообщений, топиков, прав, тредов и тп. Раньше это были одни монолитные файлы.

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

Изменения в UI

Часть изменений и доработок написал выше.

Изменение в тулбаре голосовой комнаты

Кнопки анимированные, при наведении на ПК показывают анимацию
Кнопки анимированные, при наведении на ПК показывают анимацию

Добавили выбор устройства на ПК:

Тоглам кнопкам добавили плавный переход, на фото это не будет видно.

Для мобильной версии сделали 2 вида тапа.

Короткий: Открывает меню взаимодействия с сообщениями

Длинный: позволяет выбрать несколько сообщений сразу. Удалить, переслать или ответить.

Итог

За 25 дней мы проделали огромную работу над проектом. Работали порой и до часу ночи. Все прошлые выходные провели с утра до ночи в нашем проекте.

Что сейчас:

Наши инфраструктурные сервисы готовы. Мы начинаем фазу тестирования. Активно ищем сервер в аренду с простой поддержкой и не очень дорогой. Хотели 20.04 уже запуститься, но не успели, мысли приходили на лету.

Комментарии (0)