Skip to content

Latest commit

 

History

History
155 lines (116 loc) · 17.1 KB

graduation.md

File metadata and controls

155 lines (116 loc) · 17.1 KB

Выпускной проект стажировки TopJava

Design and implement a REST API using Hibernate/Spring/SpringMVC (or Spring-Boot) without frontend.

The task is:

Build a voting system for deciding where to have lunch.

  • 2 types of users: admin and regular users
  • Admin can input a restaurant and it's lunch menu of the day (2-5 items usually, just a dish name and price)
  • Menu changes each day (admins do the updates)
  • Users can vote on which restaurant they want to have lunch at
  • Only one vote counted per user
  • If user votes again the same day:
    • If it is before 11:00 we assume that he changed his mind.
    • If it is after 11:00 then it is too late, vote can't be changed

Each restaurant provides a new menu each day.

As a result, provide a link to github repository. It should contain the code, README.md with API documentation and couple curl commands to test it (better - link to Swagger).


P.S.: Make sure everything works with latest version that is on github :)
P.P.S.: Assume that your API will be used by a frontend developer to build frontend on top of that.


error Рекомендации

Пишем выпускной проект как тестовое задание на работу

  • Не изобретай велосипедов! Грубая ошибка - пытаться сделать стандартные вещи по-своему, чаще всего криво. На проекте все должно быть единообразно! Ваш проект TopJava - сделай все МАКСИМАЛЬНО в этом стиле. Если тебе кажется, что у тебя есть лучшее решение, чем в TopJava - пишите мне в личку, я всегда открыт для улучшений.
  • Рекомендую писать проект современно: Spring Boot + Swagger/OpenAPI 3.0. Оптимально подойдет код миграции TopJava на Spring Boot в конце стажировки.

Представьте себе, что ПМ (лид, архитектор) дал вам ТЗ и некоторое время недоступен. У вас, конечно, есть много мыслей, для чего нужно приложение, как исправить ТЗ, дополнить его и сделать правильно. НО НЕ НАДО ИХ РЕАЛИЗОВЫВАТЬ В КОДЕ. Нужно сделать все максимально просто, удобно для доработок и для использования со стороны клиента (если, конечно, в ТЗ нет оговорок).

Совершенство достигнуто не тогда, когда нечего добавить, а тогда, когда нечего отнять

Антуан де Сент-Экзюпери

1: ТЗ

  • 1.1: Читай ТЗ ОЧЕНЬ внимательно, НЕ надо ничего своего туда домысливать и творчески изменять
  • 1.2: Учитывай, что пользователей может быть ОООЧЕНЬ много, а админов - МАЛО
  • 1.3: Сначала сделай основной сценарий по ТЗ. Все остальное (если очень хочется, 3 раза подумай) - потом.

2. API

  • 2.1: API продумывай с точки зрения не программиста и объектов, а с точки зрения того, кто им будет пользоваться (клиента, UI)
  • 2.1: Тщательно считайте количество запросов в вашем API для отображения нужной информации
  • 2.3: Из потребностей приложения (клиента) реализуй только очевидные сценарии. Необходимо и достаточно: ВСЕ НЕОБХОДИМОЕ для клиента и НИЧЕГО ЛИШНЕГО. Процесс творческий, приходит с опытом.
  • 2.4: Делаем REST API в соответствии с концепцией REST (url в общем имеют вид{ресурс}/{id_ресурсa}[/{подресурс}/{id_подресурсa}][параметры]). Имена ресурсов во множественном числе! Самая распространенная и грубая ошибка - не придерживаться этих простых правил.
  • 2.5: Разделение на роли я предпочитаю на уровне URL. Сразу и однозначно видно, какой API у админа, какое у юзера.
  • 2.6: На управление (CRUD) ресторанами и едой должны быть ОТДЕЛЬНЫЕ контроллеры. Не надо все, что может админ, сваливать в одну кучу!
  • 2.7: Проверьте в Swagger, что в POST и GET нет ничего лишнего, а в GET есть все необходимые данные. Например, при запросе голоса должен быть id ресторана, а при создании-редактировании ресторана не должно быть меню и еды.
  • 2.8: Profile означает, что данные принадлежат профилю пользователя. Все остальное называйте по-другому.

3: Код:

  • 5.1: Строго соблюдайте соглашения Java по именованию: пакеты ТОЛЬКО маленькими буквами, методы начинаются с маленькой буквы, классы с большой. Незнания Java Core - тестовое задание сразу в корзину.
  • 5.2 В проекте (и тестовом задании на работу), в отличие от нашего учебного topjava, оставляйте только необходимый для работы по ТЗ приложения код, ничего лишнего
    • 5.2.1: НЕ надо делать разные профили базы и работы с ней
    • 5.2.2: НЕ надо делать абстрактных контроллеров на всякий случай
    • 5.2.3: НЕ надо делать сервисов, если там нет ничего, кроме делегирования
    • 5.2.4: НЕ нужны локализация, UI, типы ошибок, Json View
  • 5.3: Название пакетов, имен классов для model/to/web стандартные (например model/domain). НЕ надо придумывать своих собственных правил
  • 5.4: Проверьте, не торчат ли из кода учебные уши TopJava, типа ProfileRestController.testUTF(), AbstractServiceTest.printResult() или закомментированные JdbcTemplate. Назначение выпускного совсем другое
  • 5.5: Вместо return ResponseEntity.ok(object) в контроллерах пишите return object. Проще!

4: Модель

  • 4.1: Историю еды и голосований делать НУЖНО (история означает, что в БД хранятся и не удаляются голоса пользователей и меню на любой день. Есть базовые вещи, которые закладываются в архитектуру приложения и неочевидные доработки к ТЗ, которых лучше не делать.
  • 4.2: Не делайте в модели объектов, которые не будут использоваться в коде (например, не надо двунаправленных связей, если достаточно однонаправленных)
  • 4.3: еще раз про hashCode/equals в Entity: не делайте в модели сравнение по полям!
  • 4.4: ORM работает с объектами. Иногда, для упрощения логики, fk_id как поля допустимы

5: Архитектура

  • 5.1: Можно:
    • или подключить DATA-REST (см.курс Spring Boot 2.x + HATEOAS). Контроллеры генерируются автоматически по репозиториям, требуется настройка ресурсов в кастомных контроллерах
    • или делать на основе миграции TopJava / кода TopJava-2

Нельзя смешивать эти подходы вместе! Я рекомендую 2-й вариант, без data-rest. Обязательно посмотрите, что у вас получилось в результате в swagger.

  • 5.2: Не размещайте бизнес-логику приложения и преобразования в TO в слое доступа к DB
  • 5.3: Не смешивайте TO и Entity вместе. Они должны быть независимыми друг от друга. На TopJava мы смотрели разные варианты c использованием TO и без. Делаем максимально просто.
  • 5.4: Репозитории
    • 5.4.1: Используйте DATA-JPA (без лишней делегации). Из сервиса/контроллера напрямую вызывайте Repository
    • 5.4.2: Если приложению в объекте требуется только его id, используйте reference (getById)
    • 5.4.3: В DATA-JPA 2.x используются Optional. Попробуйте работать с ними, это безопасный способ работать с null-значениями (используйте orElseThrow)
    • 5.4.4: Обновление в базе делается через update, даже если delete/insert сократит java-код на несколько строк
  • 5.5: Use for money in java app

6: База Данных

  • 6.1: Берите без установки (H2 или HSQLDB). Одну и в памяти! Ваше приложение должно сразу запуститься, без всяких настроек и переменных окружения
  • 6.2: Тщательно считайте количество обращений в базу на каждый запрос. Особенно при запросах от юзеров, которых очень много! Также на сложность запросов от них, чтобы не положить базу
  • 6.3: Сделайте индексы к таблицам. Попробуйте обеспечить UNIQUE (один голос пользователя в день, один уникальный пункт меню в день). Следите за порядком полей в индексе
  • 6.4: При популировании добавь записи за сегодняшний день - now(), чтобы всегда были актуальные исходные данные
  • 6.5: Поля базы case insensitive, на пишите camelStyle.
  • 6.6: Таблицы обычно именуются в единственном числе. Исключение - users, т. к. user - зарезервированное слово.
  • 6.7: date/timestamp - зарезервированное слово, лучше избегать их при именовании полей

7: Security

  • 7.1: Проверьте, станет ли код проще с @AuthenticationPrincipal (урок 11, Доступ к AuthorizedUser).
  • 7.2: Я предпочитаю четкое разделение ролей на основе URL. Для админа URL содержит /admin
  • 7.3: Еще раз - призываю не менять код TopJava

8: Кэширование необязательно, но желательно. Старайтесь сделать реализацию попроще.

  • 8.1: Тщательно продумайте, что надо кэшировать (самые частые запросы), а что нет (большие или редко запрашиваемые данные)!
  • 8.2: Проверьте соответствие ключей к кэшу (параметры кэшируемого метода) с конфигурацией (например в singleNonExpiryCache, heap=1 в кэше может содержаться только ОДНО значение).

9: Валидация данных желательна

  • 9.1: Одних аннотаций недостаточно. Должны быть @Valid/@Validation
  • 9.2: Проверяйте входные данные при create/update в контроллерах! В TopJava это ValidationUtil.checkNew()/assureIdConsistent()

10: Дополнительно

  • 10.1: По возможности сделать JUnit-тесты. Можно не делать 100% покрытие, только основные сценарии
  • 10.2: Уделяйте внимание обработке ошибок

11: readme.md:

  • 11.1: Поместите вначале readme ТЗ - будет понятно о чем он проект
  • 11.2: Если задание на English, описание пишите также на English (тоже самое относится к языку резюме: вакансия на English предполагает ваше резюме на English)
  • 11.3: Требуемые примеры curl не прячьте, а пишите здесь! Оптимально здесь должна быть ссылка на Swagger UI с креденшелами для выполнения запросов
  • 11.4: Проверяют люди с опытом в Java: не надо писать инструкций, как устанавливать Java и Maven:)

12: Git

  • Должна быть история ваших комитов с внятными комментариями. Это смотрят.
  • Не комить служенбые файлы: логи, DB, настройки IDEA и пр., это сразу - уровень Junior.
  • Все служебные файлы должны быть в .gitignore

13: Проверка

  • 13.1: Попробуйте подергать свое API по всем типичным сценариям ТЗ!
    • 13.1.2: Удобно использовать? Можно сделать проще? Например, чтобы проголосовать за ресторан залогиненному юзеру, достаточно restorauntId.
    • 13.1.3: Сколько раз пришлось его вызвать API для типичного сценария (например посмотреть рестораны с едой на сегодня)?
    • 13.1.4: Сколько запросов к базе было сделано? Можно ли сократить (например с FETCH/Graph или через кэширование)?
    • 13.1.5: Еще раз - проверьте все запросы в Sawgger, смотрите на формат запросов и данные в ответе. Все должно работать, есть все данные и нет ничего лишнего
  • 13.2: API ДОЛЖЕН соответствовать принципам REST (см. ссылки выше)
  • 13.3: ОБЯЗАТЕЛЬНО: запустите mvn test - ошибок быть не должно
  • 13.4: ОБЯЗАТЕЛЬНО: запустите приложение без всяких предварительных настроек (базы, переменных окружения, ..), лучше на другом компьютере. Приложение должно запускаться и работать!