Всем привет!
Закончилась осень, зима вступила в свои законные права, листья уже давно опали и перепутанные ветви кустарников наталкивают меня на мысли о моём рабочем Git репозитории… Но вот начался новый проект: новая команда, чистый, как только что выпавший снег репозиторий. "Тут все будет по другому" - думаю я и начинаю "гуглить" про Trunk Based Development.
Если у вас никак не получается поддерживать git flow, вам надоели кучи этих непонятных веток и правил для них, если в вашем проекте появляются ветки вида "develop/ivanov", то добро пожаловать под кат! Там я пробегусь по основным моментам Trunk Based Development и расскажу о том, как реализовать такой подход, используя Spring Boot.
Введение
Trunk Based Development (TBD) - это подход, при котором вся разработка ведется на основе единственной ветки trunk (ствол). Чтобы воплотить такой подход в жизнь, нам нужно следовать трем основным правилам:
1) Любые коммиты в trunk не должны ломать сборку.
2) Любые коммиты в trunk должны быть маленькими на столько, чтобы review нового кода не занимало более 10 минут.
3) Релиз выпускается только на основе trunk.
Договорились? Теперь давайте разбираться на примере.
Начало разработки
Initial commit
Я не придумал ничего лучше, как написать приложение "оповещатель", REST сервис которому мы передаем оповещение в виде json, а он уже оповещает конечного адресата. Для начала собираем наш проект на spring initializr. Я сделал Maven Project, язык Java 8, Spring Boot 2.4.0. Зависимости нам понадобятся следующие:
Зависимости
Название | Тип | Описание |
Spring Configuration Processor | DEVELOPER TOOLS | Generate metadata for developers to offer contextual help and "code completion" when working with custom configuration keys (ex.application.properties/.yml files). |
Validation | I/O | JSR-303 validation with Hibernate validator. |
Spring Web | WEB | Build web, including RESTful, applications using Spring MVC. Uses Apache Tomcat as the default embedded container. |
Lombok | DEVELOPER TOOLS | Java annotation library which helps to reduce boilerplate code. |
Инициализируем git репозиторий и пушим на GitHub или куда вам больше нравится. Основную ветку можно назвать по своему усмотрению: main, master или даже так и назвать - trunk, чтобы всем сразу было понятно, чем вы тут занимаетесь. Все. Посадили деревце. Теперь будем бережено его выращивать.
Первая фича
Напишем первую реализацию, которая будет отправлять сообщение на почту. Для начала опишем свойства нашего сервиса в виде ConfigurationProperties. У приложения пока будут только два свойства: sender-email - почтовый адрес отправителя и email-subject - тема письма в оповещении.
NotificationProperties
@Getter
@Setter
@Component
@Validated //говорим, что свойства должны проверяться
@ConfigurationProperties(prefix = "notification")
public class NotificationProperties {
@Email //проверяем что это почта
@NotBlank //проверяем что поле заполнено
private String senderEmail;
@NotBlank
private String emailSubject;
}
Теперь сделаем компонент, который будет отправлять оповещения на почту, делаем просто заглушу, так как скорее всего в реальности этот компонент предоставлялся бы библиотекой.
Собственно реализация для данного примера нам вообще не понадобится.
EmailSender
@Slf4j
@Component
public class EmailSender {
/**
* Отправляет сообщение на почту понарошку
*/
public void sendEmail(String from, String to, String subject, String text){
log.info("Send email\nfrom: {}\nto: {}\nwith subject: {}\nwith\n text: {}", from, to, subject, text);
}
}
Напишем простую модель для оповещения:
Notification
@Getter
@Setter
@Builder
@AllArgsConstructor
public class Notification {
private String text;
private String recipient;
}
Сервис оповещения:
NotificationService
@Service
@RequiredArgsConstructor
public class NotificationService {
private final EmailSender emailSender;
private final NotificationProperties notificationProperties;
public void notify(Notification notification){
String from = notificationProperties.getNotificationSenderEmail();
String to = notification.getRecipient();
String subject = notificationProperties.getNotificationEmailSubject();
String text = notification.getText();
emailSender.sendEmail(from, to, subject, text);
}
}
И наконец контроллер:
NotificationController
@RestController
@RequiredArgsConstructor
public class NotificationController {
private final NotificationService notificationService;
@PostMapping("/notification/notify")
public void notify(Notification notification){
notificationService.sendNotification(notification);
}
}
Ещё нам конечно понадобится тесты, без них TBD не получится. Напишем тест для NotificationService:
NotificationServiceTest
@SpringBootTest
class NotificationServiceTest {
@Autowired
NotificationService notificationService;
@Autowired
NotificationProperties properties;
@MockBean
EmailSender emailSender;
@Test
void emailNotification() {
Notification notification = Notification.builder()
.recipient("test@email.com")
.text("some text")
.build();
notificationService.notify(notification);
ArgumentCaptor<string> emailCapture = ArgumentCaptor.forClass(String.class);
verify(emailSender, times(1))
.sendEmail(emailCapture.capture(),emailCapture.capture(),emailCapture.capture(),emailCapture.capture());
assertThat(emailCapture.getAllValues())
.containsExactly(properties.getSenderEmail(),
notification.getRecipient(),
properties.getEmailSubject(),
notification.getText()
);
}
}
И для NotificationController:
NotificationControllerTest
@WebMvcTest(controllers = NotificationController.class)
class NotificationControllerTest {
@Autowired
MockMvc mockMvc;
@Autowired
ObjectMapper objectMapper;
@MockBean
NotificationService notificationService;
@SneakyThrows
@Test
void testNotify() {
ArgumentCaptor<notification> notificationArgumentCaptor = ArgumentCaptor.forClass(Notification.class);
Notification notification = Notification.builder()
.recipient("test@email.com")
.text("some text")
.build();
mockMvc.perform(post("/notification/notify")
.contentType(MediaType.APPLICATION_JSON)
.content(objectMapper.writeValueAsString(notification)))
.andExpect(status().isOk());
verify(notificationService, times(1)).notify(notificationArgumentCaptor.capture());
assertThat(notificationArgumentCaptor.getValue())
.usingRecursiveComparison()
.isEqualTo(notification);
}
}
Написали, сделали rebase, прогнали сборку с тестами и запушили в trunk - эта последовательность должна войти в привычку и делаться как можно чаще.
Для настоящих проектов я очень рекомендую делать первый коммит именно таким, чтобы он был как можно меньше и удовлетворял нашему второму правилу - code review меньше чем за 10 минут.
Профили
Начнем с самого простого и известного приема - использование профилей. Для начала вам нужно включить все ваше воображение и представить, что нам вдруг понадобилось оповещать кого-то по расписанию. Ну что же сделаем отдельный класс под эту задачу.
NotificationTask
@Component
@EnableScheduling
@RequiredArgsConstructor
public class NotificationTask {
private final NotificationService notificationService;
private final NotificationProperties notificationProperties;
@Scheduled(fixedDelay = 1000)
public void notifySubscriber(){
notificationService.notify(Notification.builder()
.recipient(notificationProperties.getSubscriberEmail())
.text("Notification is worked")
.build());
}
}
Теперь прогоним наши тесты и получим исключение для теста сервиса:
"org.mockito.exceptions.verification.TooManyActualInvocations".
Конечно, ведь в нашем тесте ожидался один вызов метода sendEmail, а получилось больше, так как теперь этот же метод вызывается в задаче.
Не порядок. Можно конечно выставить задаче initialDelay, чтобы тест успел запустится раньше чем задача, но это будет костыль. Вместо этого, как вы уже, наверное, догадались, мы применим профиль. Вынесем аннотацию @EnableScheduling в отдельную конфигурацию и добавим аннотацию @Profile, где скажем, что нужно запускать задачи всегда, кроме как в профиле "test".
SchedulingConfig
@Profile("!test")
@Configuration
@EnableScheduling
public class SchedulingConfig {}
В тестовых ресурсах, в application.yaml добавим включение профиля:
application.yaml
spring:
profiles:
active: test
notification:
email-subject: Auto notification
sender-email: robot@somecompany.com
Теперь все должно заработать. В тестах задачи по расписанию больше не запускаются, но если просто запустить приложение из main метода, то задачи будут исправно тикать.
В своей работе я, в основном, использую профили именно для тестирования, но никто вам не запретит использовать их для своих целей, главное, как мне кажется, с ними не мельчить и не создавать их много.
Используйте профили тогда, когда вам нужно включать или выключать целый слой какой-либо логики, причем на постоянной основе, т.е. вы не планируете выкинуть эту возможность когда-нибудь потом. Примерами могут служить: безопасность, мониторинг или ,как в нашем случае, задачи по расписанию.
Для более точечного управления функциями приложения лучше использовать Feature flags, но этот способ мы рассмотрим уже после нашего первого релиза. Сделали rebase, прогнали сборку с тестами и запушили в trunk.
Первый релиз
Давайте немного отвлечемся от кодирования и посмотрим, что делать с релизами. В TBD описано два способа выпускать релизы: первый из релизной ветки, второй прямо из trunk. Здесь я разберу первый способ.
Первым делом нам нужно взять коммит из которого мы будем делать релиз, это может быть как последний коммит в trunk, так и коммит который вы сделали в прошлом, все зависит от того, из какой ревизии кода вы хотите сделать релиз.
Для git выкачать прошлый коммит можно так:
git checkout <hash>
Теперь создаем новую релизную ветку, обязательно ставим метку c версией, и пушим в удаленный репозиторий.
git checkout -b Release_1.0.0
git tag 1.0.0
git push -u origin Release_1.0.0
git push origin 1.0.0
Готово! Можно разворачивать код из этой ветки в staging, а затем и в production.
Теперь мы добавим ещё парочку правил, которые будем соблюдать при работе с релизными ветками:
1) Разработчики не ведут в релизной ветке какие-либо работы.
2) Релизная ветка не сливается с trunk.
3) Если нужен Hotfix, делаем Cherry-pick из trunk и добавляем метку с минорной версией.
Таким образом, релизная ветка как бы "замораживается" и нужна только для того, чтобы выпустить из неё релиз и хранить в себе код соответствующей версии приложения. Релизные ветки можно и даже нужно удалять как только они становятся не актуальным.
Feature flags
Для второй версии нашего приложения у нас стоит задача добавить немного аудита в нашу систему: теперь каждое оповещение должно сохраняться в базу данных. Только вот проблема в том, что не известно когда на production её развернут и подготовят. Тогда, чтобы никого не ждать и не откладывать то, что можно сделать сейчас, мы обернем данную функциональность в feature flag.
Этот прием позволит нам внедрить новую фичу уже в следующем релизе, а вот включить её можно будет как только появится возможность это сделать, а в случае, если, что-то пойдет не так, фичу можно будет снова выключить.
Добавляем зависимости для взаимодействия с базой данных. БД на production у нас будет например oracle (это не особо важно для примера), а для тестов будем использовать h2.
Зависимости (БД)
<dependency>
<groupid>org.springframework.boot</groupid>
<artifactid>spring-boot-starter-data-jpa</artifactid>
</dependency>
<dependency>
<groupid>com.oracle.ojdbc</groupid>
<artifactid>ojdbc10</artifactid>
</dependency>
<dependency>
<groupid>com.h2database</groupid>
<artifactid>h2</artifactid>
</dependency>
Теперь добавим отдельный класс, где будем описывать только свойства для включения разных фич. Примем конвенцию, что все свойства в этом классе должны быть boolean. Добавим туда флаг "persistence", который будет включать и выключать сохранение оповещений в базу.
FeatureProperties
@Getter
@Setter
@Component
@ConfigurationProperties(prefix = "features.active")
public class FeatureProperties {
boolean persistence;
}
Сразу запишем в application.yaml в тестовых ресурсах features.active.persistence: on (spring сам поймет, что on==true).
Только не забудьте сначала скомпилировать проект, чтобы включилось автодополнение в свойствах приложения.
Нашу модель переделываем в Entity.
Осторожно много аннотаций!
Notification (Entity)
@Entity
@Getter
@Setter
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class Notification {
@Id
@GeneratedValue
private Long id;
private String text;
private String recipient;
@CreationTimestamp
private LocalDateTime time;
}
Добавляем репозиторий.
NotificationRepository
public interface NotificationRepository extends CrudRepository<Notification, Long> {
}
В NotificationService добавим NotificationRepository и FeatureProperties как зависимости, в конце метода notify вызовем метод репозитория save, обернув его в обычный if.
NotificationService (Feature flag)
@Service
@RequiredArgsConstructor
public class NotificationService {
private final EmailSender emailSender;
private final NotificationProperties notificationProperties;
private final FeatureProperties featureProperties;
@Nullable
private final NotificationRepository notificationRepository;
public void notify(Notification notification){
String from = notificationProperties.getSenderEmail();
String to = notification.getRecipient();
String subject = notificationProperties.getEmailSubject();
String text = notification.getText();
emailSender.sendEmail(from, to, subject, text);
if(featureProperties.isPersistence()){
notificationRepository.save(notification);
}
}
}
Забегая немного вперед, аннотация @Nullable для поля NotificationRepository нам нужна, чтобы Spring не падал с ошибкой UnsatisfiedDependencyException, если не найдет такой бин у себя в контексте.
Теперь можно запустить тесты и увидеть, что все они прошли, но если мы запустим наше приложение, то оно будет требовать указать url для базы данных и не будет запускаться.
Исправлять будем примерно так же как и для задач по расписанию. Создадим отдельную конфигурацию, где мы укажем, что автоконфигурация для базы данных должна быть исключена, если флаг features.active.persistence: off (spring сам поймет, что off==false).
DataJpaConfig
@Configuration
@ConditionalOnProperty(prefix = "features.active", name = "persistence",
havingValue = "false", matchIfMissing = true)
@EnableAutoConfiguration(exclude = {
DataSourceAutoConfiguration.class,
DataSourceTransactionManagerAutoConfiguration.class,
HibernateJpaAutoConfiguration.class})
public class DataJpaConfig {
}
Запускаем приложение с флагом features.active.persistence: off в свойствах. Приложение стартует, но не создает никаких бинов, связанных с работой базы данных.
Для того, чтобы управлять флагами в среде развертывания, можно указать spring через командную строку файл дополнительных свойств, например:
--spring.config.additional-location=file:/etc/config/features.yaml
Или передать с помощью аргументов VM, например:
-Dfeatures.active.persistence=true
Правил с флагами будет два:
1) После того, как функциональность полностью протестирована и стабильно работает, флаг этой функции нужно удалить.
2) Мест в коде, где идет ветвление по одному и тому же feature флагу, должно быть минимальное количество.
По второму правилу поясню подробнее. Если ваша новая функциональность, которую вы хотите обернуть в feature флаг, заставляет вас писать код вида: "if (flag) {…}" в нескольких местах сразу, то вам стоит задуматься либо над дизайном вашей системы, либо о приеме "ветвления по абстракции", который как раз сейчас и разберем.
Branch by Abstraction
В третьей версии настало время расширять функциональность оповещений.
Теперь с клиентской части нашего приложения в сообщениях будет приходить тип оповещения: EMAIL, SMS или PUSH. Следовательно нам необходимо реализовать два дополнительных "отправщика" сообщений, а ещё логику в самом сервисе уведомлений, которая будет определять реализацию.
Это довольно серьезная доработка, поэтому мы не хотим с ней торопится, мы хотим тщательно проработать архитектуру решения, да так, чтобы в его развитии принимало как можно больше разработчиков. Поэтому мы не будет делать в Git отдельную ветку, в которой можно было бы хранить нестабильный код, а сделаем ветку внутри программы, с помощью ветвления по абстракции.
Рецепт от Мартина Фаулера прост:
1) Выделить интерфейс для заменяемой функциональности.
2) Заменить прямой вызов реализации в клиенте на обращение к интерфейсу.
3) Создать новую реализацию, которая реализует интерфейс.
4) Подменить старую реализацию на новую.
5) Удалить старую реализацию.
Первым делом нам нужно сделать интерфейс NotificationService вместо класса, а сам класс переименовать в EmailNotificationService. В Inellij IDEA это можно провернуть с помощью рефакторинга:
1) Правой кнопкой по классу, выбрать Refactor/Extract interface…
2) Выбрать опцию "Rename original class and use interface where possible".
3) В поле "Rename implementation class to" вписываем "EmailNotificationService".
4) В "Members to from interface" нажать галочку напротив метода "notify" .
5) Нажать кнопку "Refactor".
После этого все классы должны ссылаться на интерфейс NotificationService, а рядом в пакете появится EmailNotificationService, где будет старая реализация.
Сделали rebase, прогнали сборку с тестами и запушили в trunk.
После этого можно спокойно продолжать работу уже над новой реализацией. Добавим в модель поле с типом оповещения, пусть это просто Enum.
NotificationType
public enum NotificationType {
EMAIL, SMS, PUSH, UNKNOWN
}
Так же нам нужно будет добавить два новых компонента "отправителя":
SmsSender и PushSender.
Senders
@Slf4j
@Component
public class SmsSender {
/**
* Отправляет сообщение на телефон
*/
public void sendSms(String phoneNumber, String text){
log.info("Send sms {}\nto: {}\nwith text: {}", phoneNumber, text);
}
}
@Slf4j
@Component
public class PushSender {
/**
* Отправляет push уведомления
*/
public void push(String id, String text){
log.info("Push {}\nto: {}\nwith text: {}", id, text);
}
}
Новую реализацию сервиса назовем MultipleNotificationService и для начала напишем "в лоб".
MultipleNotificationService - switch case
@Service
@RequiredArgsConstructor
public class MultipleNotificationService implements NotificationService {
private final EmailSender emailSender;
private final PushSender pushSender;
private final SmsSender smsSender;
private final NotificationProperties notificationProperties;
private final NotificationRepository notificationRepository;
@Override
public void notify(Notification notification) {
String from = notificationProperties.getSenderEmail();
String to = notification.getRecipient();
String subject = notificationProperties.getEmailSubject();
String text = notification.getText();
NotificationType notificationType = notification.getNotificationType();
switch (notificationType!=null ? notificationType : NotificationType.UNKNOWN) {
case PUSH:
pushSender.push(to, text);
break;
case SMS:
smsSender.sendSms(to, text);
break;
case EMAIL:
emailSender.sendEmail(from, to, subject, text);
break;
default:
throw new UnsupportedOperationException("Unknown notification type: " + notification.getNotificationType());
}
notificationRepository.save(notification);
}
}
Запустив тесты, мы обнаружим, что NotificationServiceTest стал падать с ошибкой:
"expected single matching bean but found 2: emailNotificationService, multipleNotificationService".
Вылечить проблему можно, например, добавлением аннотации @Primary над старой реализацией сервиса - EmailNotificationService.
@Primary сделает бин приоритетным для инъекции, но в тоже время бины с тем же типом все равно создадутся в контексте и мы сможем внедрить новую реализацию в тест.
Другой вариант - просто убрать аннотацию @Service из новой реализации, тем самым исключив её из контекста, а для теста написать отдельную конфигурацию или вообще не писать Spring тест, а написать простой unit тест, где будем создавать компоненты сами через "new".
Я воспользуюсь первым вариантом и напишу отдельный Spring тест для новой реализации.
MultipleNotificationServiceTest
@SpringBootTest
class MultipleNotificationServiceTest {
@Autowired
MultipleNotificationService multipleNotificationService;
@Autowired
NotificationProperties properties;
@MockBean
EmailSender emailSender;
@MockBean
PushSender pushSender;
@MockBean
SmsSender smsSender;
@Test
void emailNotification() {
Notification notification = Notification.builder()
.recipient("test@email.com")
.text("some text")
.notificationType(NotificationType.EMAIL)
.build();
multipleNotificationService.notify(notification);
ArgumentCaptor<string> emailCapture = ArgumentCaptor.forClass(String.class);
verify(emailSender, times(1))
.sendEmail(emailCapture.capture(),emailCapture.capture(),emailCapture.capture(),emailCapture.capture());
assertThat(emailCapture.getAllValues())
.containsExactly(properties.getSenderEmail(),
notification.getRecipient(),
properties.getEmailSubject(),
notification.getText()
);
}
@Test
void pushNotification() {
Notification notification = Notification.builder()
.recipient("id:1171110")
.text("some text")
.notificationType(NotificationType.PUSH)
.build();
multipleNotificationService.notify(notification);
ArgumentCaptor<string> captor = ArgumentCaptor.forClass(String.class);
verify(pushSender, times(1))
.push(captor.capture(),captor.capture());
assertThat(captor.getAllValues())
.containsExactly(notification.getRecipient(), notification.getText());
}
@Test
void smsNotification() {
Notification notification = Notification.builder()
.recipient("+79157775522")
.text("some text")
.notificationType(NotificationType.SMS)
.build();
multipleNotificationService.notify(notification);
ArgumentCaptor<string> captor = ArgumentCaptor.forClass(String.class);
verify(smsSender, times(1))
.sendSms(captor.capture(),captor.capture());
assertThat(captor.getAllValues())
.containsExactly(notification.getRecipient(), notification.getText());
}
@Test
void unsupportedNotification() {
Notification notification = Notification.builder()
.recipient("+79157775522")
.text("some text")
.build();
assertThrows(UnsupportedOperationException.class, () -> {
multipleNotificationService.notify(notification);
});
}
}
Сделали rebase, прогнали сборку с тестами, запушили в trunk, получили от команды по шапке за switch-case.
Реализовать логику красиво позволит шаблон "Стратегия", но тут есть проблема в том, что у всех компонент "отправителей" разные интерфейсы, собственно как скорее всего и будет в реальности, ведь обычно такие компоненты предоставляются внешними библиотеками. Решить проблему с разными интерфейсами можно с помощью шаблона "Адаптер". Расписывать подробно здесь не буду, статья всё таки о другом, но код вы можете посмотреть у меня на GitHub.
После того как сделали все красиво мы пробуем ещё раз: rebase, прогнали сборку с тестами, запушили в trunk.
На этот раз код не вызвал ни у кого негатива, и его можно включать в программу. Делать будем это с помощью все того же feature флага.
В класс с флагами добавляем новый:
boolean multipleSenders;
Над классом EmailNotificationService добавляем аннотацию с условием (ни в коем случае не удалять @Primary):
"Выключить, только, если флаг features.active.multiple-senders установлен (matchIfMissing) и равен false"
@ConditionalOnProperty(prefix = "features.active",
name = "multiple-senders",
havingValue = "false",
matchIfMissing = true)
Над MultipleNotificationService нужно добавить аннотацию с "зеркальным" условием:
"Включить, только, если флаг features.active.multiple-senders не установлен (matchIfMissing) или равен true"
@ConditionalOnProperty(prefix = "features.active",
name = "multiple-senders",
havingValue = "true",
matchIfMissing = true)
Таким образом, в тестах у нас окажутся обе реализации, а вот при запуске приложения будет работать только одна.
После того как новая версия будет обкатана, вместе с feature флагом нужно будет удалить и старую реализацию, а вот выделенный интерфейс лучше все таки оставить.
И снова rebase, прогнали сборку с тестами, запушили в trunk. Вся команда проверила ваш код и благополучно забыла, что у вас ещё есть задача по расписанию, которой не сказали с каким типом ей отправлять оповещения.
На production заметили, что оповещения по расписанию больше не запускаются, но благодаря feature флагу все сразу же откатили обратно.
Команде разработки выдали логи, она принялась за исправление, и параллельно начала обдумывать как правильно сделать Hotfix, улучшить code review и тестирование проекта, чтобы более не сталкиваться с подобными проблемами… но это уже совсем другая история, а нам пора подводить итоги.
Итоги
Trunk Based Development - отличная модель ветвления, которая наконец-то поможет вам избавится от кошмара слияния веток, позволит получить больше контроля над кодом, а команду сделать более дисциплинированной, превратит "теневое внедрение" из просто интересного приема в обыденность.
Trunk Based Development - очень гибкая методология, у неё есть несколько вариаций, из которых вы сможете выбрать наиболее подходящий вариант, и ,конечно, для её применения не обязательно использовать Spring Boot, но надеюсь я смог показать вам, что с ним это просто и удобно.
На этом всё, внизу будут все ссылки из статьи, спасибо за внимание!
gsaw
Сложно как то. Получается архитектура софта такая, потому, что контроль версий такой. И контроль версиями без Спрингс сильно усложняется, так как придется городить фичофлаги, придумывать механизм управления компонентами. И так далее.
Короче сплошные зависимости. Ещё и в коде будут кучи проверок, активна фича или нет, причем нет гарантии, что где то такую проверку не забыли вставить и вообще все равно придется менять стабильный код, встраивая ветвление. Где гарантия, что не напартачили? Значит усложняется тестирование, старого и нового кода перед релизом. Клиенты будут рады.
Да ещё требование, что бы комитты были небольшими. Такого не бывает, если только баги какие то. Для чего то большего все равно придется бранчи создавать и вот мы мы тут снова.
zizitop13 Автор
Здравствуйте.
Все сложно когда в первый раз.
Архитектура такая из-за современных требований, например по проведению A/B тестирование. Методология ветвления просто подстроилась под новые требования и новую архитектуру.
Без спринг обойтись можно, а вот без инверсии зависимостей наверное нет, но не спринг один предоставляет CDI контейнер.
Гарантий кроме вас самих вам ни кто не даст, делайте код ревью пишите тесты.
Возможность создания небольших коммитов и достигается за счёт поэтапного создания нового функционала, т.е. вы закидываете в транк тот код который ещё не до конца доделан, но он ни чего не ломает.