Upgrade to Pro — share decks privately, control downloads, hide ads and more …

Spring Boot and Cloud Workshop at JPoint 2019

Spring Boot and Cloud Workshop at JPoint 2019

Kirill Tolkachev

April 08, 2019
Tweet

More Decks by Kirill Tolkachev

Other Decks in Technology

Transcript

  1. All your classes All your pages and xmls Regular web.xml

    with spring dispatcher servlet, spring listener and security filter name Security filters, users and roles Spring MVC beans Application beans, Import security.xml The project structure
  2. web.xml <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <filter> <filter-name>springSecurityFilterChain</filter-name> <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class> </filter> <filter-mapping> <filter-name>springSecurityFilterChain</filter-name>

    <url-pattern>/*</url-pattern> </filter-mapping> <servlet> <servlet-name>mvc-dispatcher</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>mvc-dispatcher</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping>
  3. Deployment →Use maven/gradle to build a war from that project

    →Copy it to webapps directory of tomcat
  4. How does it work? User type in his browser: http://localhost:8080/myApp

    create Spring ContextLoaderListener Start tomcat start applicationContext In bootsrap In bootsrap Create filter mapping Create security filter login Create spring dispatcher servlet Delegate to Controller
  5. § 8.2.4 Shared libraries / runtimes pluggability The ServletContainerInitializer class

    is looked up via the jar services API. For each application, an instance of the ServletContainerInitializer is created by the container at application startup time. The framework providing an implementation of the ServletContainerInitializer MUST bundle in the META-INF/services directory of the jar file a file called javax.servlet.ServletContainerInitializer, as per the jar services API, that points to the implementation class of the ServletContainerInitializer. 3.+
  6. § 8.2.4 Shared libraries / runtimes pluggability The ServletContainerInitializer’s onStartup

    method get's a Set of Classes that either extend / implement the classes that the initializer expressed interest in or if it is annotated with any of the classes specified via the @HandlesTypes annotation. 3.+
  7. Как? SPI tomcat |→ get javax.servlet.ServletContainerInitializer impl via SPI |→

    get all WebApplicationInitializer classes (from @HandlesTypes) org.springframework.web.WebApplicationInitializer
  8. Как? SPI tomcat |→ get javax.servlet.ServletContainerInitializer impl via SPI |→

    get all WebApplicationInitializer classes (from @HandlesTypes) |→ call ServletContainerInitializer.onStartup(classes,servletCtx) according to @Order order
  9. @HandlesTypes(WebApplicationInitializer.class) public class SpringServletContainerInitializer implements ServletContainerInitializer { @Override public void

    onStartup(Set<Class<?>> webAppInitializerClasses, ServletContext servletContext) ... AnnotationAwareOrderComparator.sort(initializers); ...
  10. @HandlesTypes(WebApplicationInitializer.class) public class SpringServletContainerInitializer implements ServletContainerInitializer { @Override public void

    onStartup(Set<Class<?>> webAppInitializerClasses, ServletContext servletContext) ... AnnotationAwareOrderComparator.sort(initializers); ... initializers.forEach(initializer -> initializer.onStartup(ctx)); }
  11. public class WebStarter implements WebApplicationInitializer { @Override public void onStartup(ServletContext

    servletContext) throws ServletException { AnnotationConfigWebApplicationContext webContext = new AnnotationConfigWebApplicationContext(); webContext.register(WebConfig.class); ServletRegistration.Dynamic servlet = servletContext.addServlet("dispatcher", new DispatcherServlet(webContext)); servlet.setLoadOnStartup(1); AnnotationConfigWebApplicationContext appContext = new AnnotationConfigWebApplicationContext(); appContext.register(AppConfig.class); servletContext.addListener(new ContextLoaderListener(appContext)); servlet.addMapping("/*"); }
  12. Как выглядит наш build.gradle plugins { id "org.springframework.boot" version "2.1.4.RELEASE"

    } dependencies { compile 'org.springframework.boot:spring-boot-starter-web' } Добавляем зависимости
  13. Как выглядит наш build.gradle plugins { id "org.springframework.boot" version "2.1.4.RELEASE"

    } dependencies { compile 'org.springframework.boot:spring-boot-starter-web' } Automatic Dependency Management No Version
  14. Как выглядит наш build.gradle plugins { id "org.springframework.boot" version "2.1.4.RELEASE"

    } dependencies { compile 'org.springframework.boot:spring-boot-starter-web' } dependencyManagement { imports { mavenBom 'org.springframework.cloud:spring-cloud-dependencies:Greenwich.RELEASE' } }
  15. Анатомия SpringBoot Jar jar |- META-INF |- BOOT-INF |- libs

    |- classes |- org |- springframework |- boot
  16. java -jar myapp.jar myapp.jar |- META-INF |- BOOT-INF |- libs

    |- classes |- org |- springframework |- boot MANIFEST.MF ... Main-Class: ??? ...
  17. Анатомия SpringBoot Jar myapp.jar |- META-INF |- BOOT-INF |- libs

    |- classes |- org |- springframework |- boot MANIFEST.MF ... Spring-Boot-Version: 1.5.3.RELEASE Implementation-Vendor: Pivotal Software, Inc. Main-Class: org.springframework.boot.loader.JarLauncher Start-Class:ru….ripper.OurMainClass Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ ...
  18. myapp.jar |- META-INF |- BOOT-INF |- libs |- classes |-

    org |- springframework |- boot Анатомия SpringBoot Jar MANIFEST.MF ... Spring-Boot-Version: 1.5.3.RELEASE Implementation-Vendor: Pivotal Software, Inc. Main-Class: org.springframework.boot.loader.JarLauncher Start-Class:ru….ripper.OurMainClass Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ ...
  19. MANIFEST.MF ... Spring-Boot-Version: 1.5.3.RELEASE Implementation-Vendor: Pivotal Software, Inc. Main-Class: org.springframework.boot.loader.JarLauncher

    Start-Class:ru….ripper.OurMainClass Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ ... myapp.jar |- META-INF |- BOOT-INF |- libs |- classes |- org |- springframework |- boot Анатомия SpringBoot Jar JarLauncher
  20. Как выбирается mainClass если его не указать Spring boot plugin

    сканирует проект – ищет мейны • если есть только один – то это он :) • если больше одного – смотрит где стоит @SpringBootApplication и выбирает его • если @SpringBootApplication нет или >1 – ошибка о множественных main при сборке
  21. Хочу executable jar: Maven <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration>

    <executable>true</executable> </configuration> </plugin> </plugins> </build> springBoot { executable = true } Gradle bootJar { launchScript() } Spring boot 1.x.x Spring boot 2.x.x
  22. Внутренний мир executable Jar • Бабушка хочет чтобы жмя и

    работало • windows “click click” • $ ./app.jar • ...
  23. Как выглядит Jar? Script 0xf4ra -> а это ZIP! jar

    archive Начало файла Начало zip архива
  24. Вот контекст! @SpringBootApplilcation class App { public static void main(String[]

    args) { ApplicationContext context = SpringApplication.run(App.class,args); } }
  25. Это я решаю какой контекст создать Я до создания контекста

    ещё много всего делаю, но об этом потом. SpringApplication
  26. Задание 1. Печатать предупреждение при вызове @Deprecated 2. Печатать время

    исполнения методов @Benchmark 3. Отправить email на адрес alarm.email если вызвали @Deprecated или @Benchmark
  27. Что за spring.factories ? § 44.1 Understanding auto-configured beans Under

    the hood, auto-configuration is implemented with standard @Configuration classes. Additional @Conditional annotations are used to constrain when the auto-configuration should apply. Usually auto-configuration classes use @ConditionalOnClass and @ConditionalOnMissingBean annotations. This ensures that auto-configuration only applies when relevant classes are found and when you have not declared your own @Configuration. You can browse the source code of spring-boot-autoconfigure to see the @Configuration classes that we provide (see theMETA-INF/spring.factories file).
  28. @SpringBootApplication @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented @Inherited @SpringBootConfiguration @EnableAutoConfiguration @ComponentScan(excludeFilters = {

    @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class), @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) }) public @interface SpringBootApplication { …
  29. @SpringBootApplication @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented @Inherited @SpringBootConfiguration @EnableAutoConfiguration @ComponentScan(excludeFilters = {

    @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class), @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) }) public @interface SpringBootApplication { …
  30. @EnableAutoConfiguration @Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented @Inherited @AutoConfigurationPackage @Import({EnableAutoConfigurationImportSelector.class}) public @interface EnableAutoConfiguration

    { String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration"; Class<?>[] exclude() default {}; String[] excludeName() default {}; } ImportSelector
  31. SpringFactoriesLoader static <T> List<T> loadFactories( Class<T> factoryClass, ClassLoader cl )

    static List<String> loadFactoryNames( Class<?> factoryClass, ClassLoader cl )
  32. CacheConfigurations private static final Map<CacheType, Class<?>> MAPPINGS; static { Map<CacheType,

    Class<?>> mappings = new HashMap<CacheType, Class<?>>(); mappings.put(CacheType.GENERIC, GenericCacheConfiguration.class); mappings.put(CacheType.EHCACHE, EhCacheCacheConfiguration.class); mappings.put(CacheType.HAZELCAST, HazelcastCacheConfiguration.class); mappings.put(CacheType.INFINISPAN, InfinispanCacheConfiguration.class); mappings.put(CacheType.JCACHE, JCacheCacheConfiguration.class); mappings.put(CacheType.COUCHBASE, CouchbaseCacheConfiguration.class); mappings.put(CacheType.REDIS, RedisCacheConfiguration.class); mappings.put(CacheType.CAFFEINE, CaffeineCacheConfiguration.class); addGuavaMapping(mappings); mappings.put(CacheType.SIMPLE, SimpleCacheConfiguration.class); mappings.put(CacheType.NONE, NoOpCacheConfiguration.class); MAPPINGS = Collections.unmodifiableMap(mappings); }
  33. Где то внутри CacheAutoConfiguration for (int i = 0; i

    < types.length; i++) { Imports[i] = CacheConfigurations.getConfigurationClass(types[i]); } return imports;
  34. Conditional и друзья @ConditionalOnBean @ConditionalOnClass @ConditionalOnCloudPlatform @ConditionalOnExpression @ConditionalOnJava @ConditionalOnJndi @ConditionalOnMissingBean

    @ConditionalOnMissingClass @ConditionalOnNotWebApplication @ConditionalOnProperty @ConditionalOnResource @ConditionalOnSingleCandidate @ConditionalOnWebApplication ...
  35. Паззлер @Configuration @ConditionalOnWinterIsHere public class UndeadArmyConfiguration { @Bean public WinterUndeadArmy

    cursedArmy() { return new WinterUndeadArmy(); } @Bean @ConditionalOnWinterIsHere public DragonGlassFactory dragonGlassFactory() { return new DragonGlassFactory(); }
  36. ConditionalOnPuzzler @Configuration public class КонфигурацияКазни { @Bean @ConditionalOnClass ({Мыло.class, Веревка.class})

    @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни виселицы() { return new ФабрикаВиселиц( "..."); } }
  37. ConditionalOnPuzzler @Configuration public class КонфигурацияКазни { @Bean @ConditionalOnClass ({Мыло.class, Веревка.class})

    @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни виселицы() { return new ФабрикаВиселиц( "..."); } @Bean @ConditionalOnClass ({Стул.class, Ток.class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни cтулья() { return new ФабрикаЭлектрическихСтульев( "вж вж"); } }
  38. ConditionalOnPuzzler @Configuration public class КонфигурацияКазни { @Bean @ConditionalOnClass ({Мыло.class, Веревка.class})

    @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни виселицы() { return new ФабрикаВиселиц( "..."); } @Bean @ConditionalOnClass ({Стул.class, Ток.class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни cтулья() { return new ФабрикаЭлектрическихСтульев( "вж вж"); } @Bean @ConditionalOnClass ({Гильотина.class, ХорошееНастроение. class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни гильотины() { return new ФабрикаГильотин( "хрусть хрусть"); } }
  39. ConditionalOnPuzzler @Configuration public class КонфигурацияКазни { @Bean @ConditionalOnClass ({Мыло.class, Веревка.class})

    @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни виселицы() { return new ФабрикаВиселиц( "..."); } @Bean @ConditionalOnClass ({Стул.class, Ток.class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни cтулья() { return new ФабрикаЭлектрическихСтульев( "вж вж"); } @Bean @ConditionalOnClass ({Гильотина.class, ХорошееНастроение. class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни гильотины() { return new ФабрикаГильотин( "хрусть хрусть"); } } 1. ClassDefNotFound ? 2. Так вообще нельзя, не компилируется 3. Будет работать 4. Будет отлично работать
  40. ConditionalOnPuzzler @Configuration public class КонфигурацияКазни { @Bean @ConditionalOnClass ({Мыло.class, Веревка.class})

    @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни виселицы() { return new ФабрикаВиселиц( "..."); } @Bean @ConditionalOnClass ({Стул.class, Ток.class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни cтулья() { return new ФабрикаЭлектрическихСтульев( "вж вж"); } @Bean @ConditionalOnClass ({Гильотина.class, ХорошееНастроение. class}) @ConditionalOnMissingBean ({ФабрикаКазни. class}) public ФабрикаКазни гильотины() { return new ФабрикаГильотин( "хрусть хрусть"); } } 1. ClassDefNotFound ? 2. Так вообще нельзя, не компилируется 3. Будет работать 4. Будет отлично работать
  41. OnPropertyCondition @Conditional(OnPropertyCondition.class) public @interface ConditionalOnProperty { String[] value() default {};

    String prefix() default ""; String[] name() default {}; String havingValue() default ""; boolean matchIfMissing() default false; boolean relaxedNames() default true; } Что если нужно изменить?
  42. public class WebStarter implements WebApplicationInitializer { @Override public void onStartup(ServletContext

    servletContext) throws ServletException { AnnotationConfigWebApplicationContext webContext = new AnnotationConfigWebApplicationContext(); webContext.register(WebConfig.class); ServletRegistration.Dynamic servlet = servletContext.addServlet("dispatcher", new DispatcherServlet(webContext)); servlet.setLoadOnStartup(1); AnnotationConfigWebApplicationContext appContext = new AnnotationConfigWebApplicationContext(); appContext.register(AppConfig.class); servletContext.addListener(new ContextLoaderListener(appContext)); servlet.addMapping("/*"); }
  43. public class WebStarter implements WebApplicationInitializer { @Override public void onStartup(ServletContext

    servletContext) throws ServletException { AnnotationConfigWebApplicationContext webContext = new AnnotationConfigWebApplicationContext(); webContext.register(WebConfig.class); ServletRegistration.Dynamic servlet = servletContext.addServlet("dispatcher", new DispatcherServlet(webContext)); servlet.setLoadOnStartup(1); AnnotationConfigWebApplicationContext appContext = new AnnotationConfigWebApplicationContext(); appContext.register(AppConfig.class); servletContext.addListener(new ContextLoaderListener(appContext)); servlet.addMapping("/*"); }
  44. public class WebStarter implements WebApplicationInitializer { @Override public void onStartup(ServletContext

    servletContext) { SpringApplication.run(RipperApplication.class); } } @SpringBootApplication public class RipperApplication { public static void main(String[] args) { SpringApplication.run(RipperApplication.class, args); } }
  45. public class WebStarter implements WebApplicationInitializer{ @Override public void onStartup(ServletContext servletContext)

    throws ServletException { SpringApplication.run(RipperApplication.class); } } Tomcat в Tomcat`е
  46. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder app) { return app.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } }
  47. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder app) { return app.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } }
  48. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder app) { return app.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } } Работает по разному
  49. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder app) { return app.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } }
  50. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder a) { return application.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } } Никогда не запустится в tomcat Только java -jar
  51. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder a) { return application.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } } Без main не соберется см. maven/gradle плагины Обязательно
  52. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder a) { return application.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } } Опционально
  53. § 85.1 Create a deployable war file @SpringBootApplication public class

    Application extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder a) { return application.sources(Application.class); } public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } } Никогда не запустится в embedded режиме Только в Tomcat контейнере Опционально
  54. Анатомия War .war |- META-INF |- WEB-INF |- lib |-

    classes |- org/springframework/boot/loader |- WarLauncher and friends
  55. А как убрать tomcat из tomcat? If you are using

    a version of Gradle that supports compile only dependencies (2.12 or later), you should continue to use providedRuntime. Among other limitations, compileOnly dependencies are not on the test classpath so any web-based integration tests will fail.
  56. Анатомия War .war |- META-INF |- WEB-INF |- lib |-

    lib-provided |- classes |- org/springframework/boot/loader |- WarLauncher and friends
  57. Анатомия SpringBoot Jar/War jar |- META-INF |- BOOT-INF |- libs

    |- classes |- org/springframework/boot/loader |- JarLauncher and friends war |- META-INF |- WEB-INF |- lib |- lib-provided |- classes |- org/springframework/boot/loader |- WarLauncher and friends
  58. @RestController("/examine") Это имя spring bean, не http path И это

    даже может работать: • Если нет ни одного контроллера с путём (всегда будет вызываться) • Если потом указали полный путь в @RequestMapping
  59. @RequestMapping и его братья Управляет маршрутизацией запросов в метод контроллера.

    Маршрутизирует по: • name/value – сам http path • method – http method • params • headers
  60. @RequestMapping и его братья @PostMapping("/examine") public CheckedExam ex`( @RequestBody SolvedExam

    title) { } @RequestMapping (path = "/examine", method = POST) public CheckedExam ex2( @RequestBody SolvedExam title) { } @RequestMapping (path = "/examine", method = POST, headers = {"my-app=appid"}) public CheckedExam ex3( @RequestBody SolvedExam title) { } @GetMapping(path = "/examine", headers = {"content-type=application/json"}) public CheckedExam ex3() { } @GetMapping(path = "/examine", produces = MimeTypeUtils. APPLICATION_JSON_VALUE) public CheckedExam ex3() { }
  61. @RequestMapping и его братья @PostMapping("/examine") public CheckedExam ex`( @RequestBody SolvedExam

    title) { } @RequestMapping (path = "/examine", method = POST) public CheckedExam ex2( @RequestBody SolvedExam title) { } @GetMapping(path = "/examine", headers = {"content-type=application/json"}) public CheckedExam ex3() { } @GetMapping(path = "/examine", produces = APPLICATION_JSON_VALUE) public CheckedExam ex3() { } @GetMapping(path = "/examine", produces = APPLICATION_JSON_VALUE) public CheckedExam ex4() { }
  62. @RequestMapping и его братья @PostMapping("/examine") public CheckedExam ex`( @RequestBody SolvedExam

    title) { } @RequestMapping (path = "/examine", method = POST) public CheckedExam ex2( @RequestBody SolvedExam title) { } @GetMapping(path = "/examine", headers = {"content-type=application/json"}) public CheckedExam ex3() { } @GetMapping(path = "/examine", produces = APPLICATION_JSON_VALUE) public CheckedExam ex3() { } @GetMapping(path = "/examine", produces = APPLICATION_JSON_VALUE) public CheckedExam ex4() { }
  63. @RequestMapping и его братья //curl -XPOST /examine -d '{ "title":

    {..} }' @PostMapping("/examine") public CheckedExam ex`( @RequestBody SolvedExam exam) { } //curl -XGET /examine/ mytitle @GetMapping("/examine/{title}") public CheckedExam ex3( @PathVariable String title) { } //curl -XGET /examine/mytitle ?debug=true @GetMapping("/examine/{title}") public CheckedExam ex3( @PathVariable String title, @RequestParam Boolean debug) { }
  64. Controllers endpoints in logs since Spring 5 logging.level: //print mapped

    controllers org.springframework.web.servlet.mvc.method.annotation: TRACE web: DEBUG //print requests to console
  65. Задание 0 Напишем стартер DoD: 1. Все контроллеры помеченые @FrontendController

    аннотацией должны обворачивать возвращаемый результат в дополнительный Json { "result": { оригинальный_json } } 2. https://github.com/lavcraft/spring-boot-and-cloud-ripper-2019 3.
  66. { "География":5, "Физика" :3 } Физика Математика Теология Не грешно

    ли лезть в атом? Теорема Пифагора Неравенство Коши E=? Количество вопросов Количество вопросов Экзаминатор
  67. С чего начинают строить микросервис →Контроллеры →Сервисы →Dao →Model –

    поскольку модель используется на всех слоях, пожалуй лучше начинать с неё
  68. in: { "student": "Борисов Евгений", "title": "Экзамен на JokerConf", "sections":

    [{ "title": "Java", "description": "простые вопросы по java", "exercises": [ { "question": "разница между spring string и swing", "answer": "без spring`а не обойтись" }, { "question": "разница между final finally finalize", "answer": "за finalize отрывают руки" } ]}, { "title": "философия", "description": "сложные вопросы по философии", "exercises": [{ "question": "В чем смысл жизни", "answer": "42" }, { "question": "Что раньше, яйцо или курица", "answer": "петух" }] }] } out: { "mark": 99, "title": "Экзамен на JokerConf", "sections": [{ "title": "Java", "description": "простые вопросы по java", "exercises": [ { "question": "разница между spring string и swing", "answer": "без spring`а не обойтись" }, { "question": "разница между final finally finalize", "answer": "за finalize отрывают руки" } ]}, { "title": "философия", "description": "сложные вопросы по философии", "exercises": [{ "question": "В чем смысл жизни", "answer": "42" }, { "question": "Что раньше, яйцо или курица", "answer": "петух" }] }] }
  69. Data model - задание 1.1* Придумайте архитектуру модели, которая позволит

    написать Сервис: - принимающий на вход объект SolvedExam - отдающий CheckedExam с оценкой - Должны проходить тесты @Test public void checkExamineContract() throws Exception { ... mockMvc.perform(post("/examine")) ... } Use @RestController and @RequestMapping > @*Mapping Luke
  70. Data model - задание 1.1* Придумайте архитектуру модели, которая позволит

    написать Сервис: - принимающий на вход объект SolvedExam - отдающий CheckedExam с оценкой - Смотрите в src/test/resources/*.json - Должны проходить тесты @Test public void checkExamineContract() throws Exception { ... mockMvc.perform(post("/examine")) ... } Use @RestController and @RequestMapping > @*Mapping Luke
  71. Lombok – composition / delegate and friends Annotation Processor Работает

    на этапе компиляции Генерит код, чтобы мы не писали Делает джаву более похожим на нормальный язык
  72. Про lombok, модели и не только @Data – POJO (@Getter,

    @Setter, @ToString, @EqualsAndHashcode) @Value – immutable POJO @AllArgumentConstruct(onConstructor = @_(@Autowired))
  73. Про lombok, модели и не только @Data – POJO (@Getter,

    @Setter, @ToString, @EqualsAndHashcode) @Value – immutable POJO @AllArgumentConstruct(onConstructor = @_(@Autowired)) @RequireArgumentConstructor /@NoArgumentConstructor @Builder / @Singular @Delegate – примерно как в груви @SneakyThrows @Slf4j / @Log4j / …
  74. Как Джексон пишет в джейсон 1. Использует Java Getters a.

    Методы начинающиеся с Get 2. @JsonIgnore – игнорируем Getter 3. Нет ни одного Getter – падает
  75. Как Джексон пишет в объект 1. Нет конструктора с @ConstructProperies

    → выставляет через Setters → если нет Setter и есть Getter – выставляет напрямую в филды → это все только при наличии пустого конструктора 2. Есть конструктор с @ConstructProperies → выставляет значение через него 3. После зачем то вызовет все геттеры * Но если есть то @JsonIgnore всё по другому * Это поведение по умолчанию
  76. → Если вы настраиваете руками: ObjectMapper mapper = new ObjectMapper()

    .registerModule( new ParameterNamesModule()) .registerModule( new Jdk8Module()) .registerModule( new JavaTimeModule()); → Или так: mapper.findAndRegisterModules(); Не забыть зависимости: compile 'com.fasterxml.jackson.module:jackson-module-parameter-names' compile 'com.fasterxml.jackson.datatype:jackson-datatype-jdk8' compile 'com.fasterxml.jackson.datatype:jackson-datatype-jsr310' C последних версий есть поддержка Java8
  77. Как Spring MVC работает с Джексоном Напишите метод который принимает

    json и класс, а возвращает объект данного класса Пробуем использовать objectMapper Jackson2
  78. Кто использует Jackson в мире Spring MVC 1. ClassLoader 2.

    Dispatcher Servlet 3. Tomcat 4. BeanPostProcessor
  79. Кто использует Jackson в мире Spring MVC 1. ClassLoader 2.

    Dispatcher Servlet 3. Tomcat 4. BeanPostProcessor
  80. Задание 2 Микросервис – Теологии 1. Добавлять вопросы по теологии

    … (CRUD) 2. Получать нужное количество рандомальных вопросов 3. Тест должен проходить @Test public void should_return_random_exercise() throws Exception { mvc.perform(get("/exercise/random?count=3") .contentType( APPLICATION_JSON) ) … Use JpaRepository and @RequestMapping Luke
  81. { "География":5, "Физика" :3 } Физика Математика Теология Не грешно

    ли лезть в атом? Теорема Пифагора Неравенство Коши ваы Количество вопросов Количество вопросов Экзаминатор Person p = restTemplate.getForObject("/{name}/details", Person.class, name); Для связи сервисов используем RestTemplate
  82. Конфликт портов — relaxed properties 1. server.port в application.properties 2.

    export SERVER_PORT=8081 3. java -jar --server.port=8081 4. SPRING_APPLICATION_JSON = '{"server":{"port":8081}}'
  83. in: { "student": "Борисов Евгений", "title": "Экзамен на JokerConf", "sections":

    [{ "title": "Java", "description": "простые вопросы по java", "exercises": [ { "question": "разница между spring string и swing", "answer": "без spring`а не обойтись" }, { "question": "разница между final finally finalize", "answer": "за finalize отрывают руки" } ] }, { "title": "философия", "description": "сложные вопросы по философии", "exercises": [{ "question": "В чем смысл жизни", "answer": "42" }, { "question": "Что раньше, яйцо или курица", "answer": "петух" }] }] }
  84. Дальше пишем Entity + Repository Which repositories get exposed by

    defaults? Name Description DEFAULT Exposes all public repository interfaces but considers @Repository/@RestResource’s `exported flag ALL Exposes all repositories independently of type visibility and annotations ANNOTATION Only repositories annotated with @Repository/@RestResource are exposed, unless their exported flag is set to false VISIBILITY Only public repositories annotated are exposed
  85. Как поменять стратегию @Component public class MyWebConfiguration extends RepositoryRestConfigurerAdapter {

    @Override public void configureRepositoryRestConfiguration( RepositoryRestConfiguration config) { config.setRepositoryDetectionStrategy( ALL); } }
  86. Как задать URL @Configuration class CustomRestMvcConfiguration { @Bean public RepositoryRestConfigurer

    repositoryRestConfigurer() { return new RepositoryRestConfigurerAdapter() { @Override public void configureRepositoryRestConfiguration( RepositoryRestConfiguration config) { configuration.setBasePath( "/api") } }; } }
  87. Как задать URL @Configuration class CustomRestMvcConfiguration { @Bean public RepositoryRestConfigurer

    repositoryRestConfigurer() { return new RepositoryRestConfigurerAdapter() { @Override public void configureRepositoryRestConfiguration( RepositoryRestConfiguration config) { configuration.setBasePath( "/api") } }; } } Или в том же RepositoryRestConfigurerAdapter прописать в application.properties - spring.data.rest.base-path=/api
  88. Spring Data Rest official supports →Spring Data JPA →Spring Data

    MongoDB →Spring Data Neo4j →Spring Data GemFire →Spring Data Cassandra
  89. @RepositoryRestResource @RepositoryRestResource (collectionResourceRel = "people", path = "people") public interface

    PersonRepository extends MongoRepository<Person, String> { @RestResource(path = "byname") List<Person> findByLastName(@Param( "name") String name); }
  90. URL’s и методы localhost:8080/exercise – список всех (GET) localhost:8080/exercise/1 –

    дай человека с айдишником 1 (GET) localhost:8080/exercise/1 – стереть с айдишником 1 (DELETE) localhost:8080/exercise/1 – заменить с айдишником 1 (PUT) localhost:8080/exercise/1 – проапдэйтить с айдишником 1 (PATCH) localhost:8080/exercise/search/findByName?name=Lanister
  91. { "География":5, "Физика" :3 } Физика Математика Теология Не грешно

    ли лезть в атом? Теорема Пифагора Неравенство Коши ваы Количество вопросов Количество вопросов Экзаминатор
  92. Задание 3 Связать сервисы Экзаменатора и Предметы • Запустить и

    проверить самим • Должен отдаваться запрос с ответами другого сервиса { "География":5, "Физика" :3 }
  93. --server.port Конфликт портов решается с помощью relaxed properties export SERVER_PORT=8081

    java -jar --server.port=8081 SPRING_APPLICATION_JSON = '{"server":{"port":8081}}'
  94. RestTemplate • you can create it even with new /

    or inject configured bean • postForEntity / postForObject/ getForEntity / getForObject restTemplate.getForObject( "http://"+serviceName+"/exercice/random?count=" + number, Exercice[].class); • exchange restTemplate.exchange(url.toString(), HttpMethod.POST, entity, String.class);
  95. Подход №1 - общий код Зачем разбивать микросервис на модули?

    → Java API для сервисов → фиксация контракта на уровне зависимостей языка → утилитарная функция, избавления от бойлерплейта
  96. Подход №1 - общий код project |- project-sdk |- project-app

    |- project-api |- project-domain |- project-... Пфф
  97. Подход №1 - общий код project |- project-sdk |- project-app

    |- project-api |- project-domain |- project-... Пфф KISS
  98. Обратимся к истории → REST подход ◦ анархия и рекомендации

    → RPC подход ◦ jax-ws/jax-rpc ◦ corba ◦ json rpc ◦ ◦
  99. Обратимся к истории → REST подход ◦ анархия и рекомендации

    → RPC подход ◦ jax-ws/jax-rpc ◦ corba ◦ json rpc ◦ thrift ◦ protobuf/grpc ◦ etc
  100. Задание 5 • пишем SDK для вызова микросервиса по теологии

    • и тесты для него • тестируем контракт • тестируем взаимодействие ◦ WireMock/Spring MockServer ◦ Spring Boot Test ◦ Context Scanning behaviour Тест SDK Тест Контракта Тест Взаимодействия
  101. { "География":5, "Физика" :3 } Физика Математика Теология Не грешно

    ли лезть в атом? Теорема Пифагора Неравенство Коши Энтропия не? Количество вопросов Количество вопросов Экзаминатор
  102. { "География":5, "Физика" :3 } Физика Математика Теология Теорема Пифагора

    Неравенство Коши ваы Количество вопросов Количество вопросов Экзаминатор Бог Не грешно ли лезть в атом?
  103. Матмематика – сервис • Сделать новый модуль • Реализовать метод

    "/exercise/random" • Возвращать список автогенерируемых упражнение (как в Теологии)
  104. Как воткнуть в Examinator math-service? exercises: urls: "theology" : "http://localhost:8080/"

    "math" : "http://localhost:8082/" Потому что на 8081 Examinator
  105. Как воткнуть в Examinator math-service? exercises: urls: "theology" : "http://localhost:8080/"

    "math" : "http://localhost:8082/" "examinator": "http://localhost:8081/" "..." : "http://localhost:8083/" Service Registry в application.yml
  106. { "География":5, "Физика" :3 } Физика Математика Количество вопросов Количество

    вопросов Экзаминатор Экзаминатор Теология Discovery Server
  107. { "География":5, "Физика" :3 } Физика Математика Количество вопросов Количество

    вопросов Экзаминатор Экзаминатор Теология Discovery Server Eureka Clients Eureka Server
  108. Eureka Server and Client compile 'org.springframework.cloud:spring-cloud-starter-eureka-server' compile 'org.springframework.cloud:spring-cloud-starter-eureka' С Greenwich

    релиза compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-client' compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-server' Spring Boot 1.x.x Spring Boot 2.x.x
  109. Собственный Eureka Server @EnableEurekaServer @SpringBootApplication public class EurekaServer { public

    static void main(String[] args) { SpringApplication.run(EurekaServer.class, args); } } server.port: 8761 spring.application.name: registry eureka: client.register-with-eureka: false server.enable-self-preservation: false
  110. Переводим на Eureka • Создаём модуль eureka-server compile 'org.springframework.cloud:spring-cloud-starter-eureka-server' compile

    'org.springframework.cloud:spring-cloud-starter-eureka' Or compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-server' compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-client'
  111. Переводим на Eureka • Создаём модуль eureka-server compile 'org.springframework.cloud:spring-cloud-starter-eureka-server' compile

    'org.springframework.cloud:spring-cloud-starter-eureka' Or compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-server' compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-client' • Интегрируем клиента (examinator-service, theology-service, math-service) compile 'org.springframework.cloud:spring-cloud-starter-eureka' Or compile 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-client'
  112. Переводим на Eureka @EnableDiscoveryClient vs @EnableEurekaClient eureka.client.service-url.default-zone: http://127.0.0.1:8761/eureka/ spring.application.name: EXAMINATOR

    @Никаких Аннотация не нужно в 2хх eureka.client.service-url.default-zone: http://127.0.0.1:8761/eureka/ spring.application.name: EXAMINATOR Spring Boot 1.x.x Spring Boot 2.x.x
  113. @EnableDiscoveryClient Будет включен по умолчанию при добавлении стартера c Edgware

    spring-cloud release train application.yml: server.port: 8081 spring.application.name: examinator eureka.client.service-url.defaultZone:http://localhost:8671/eureka
  114. @EnableEurekaServer Будет включен по умолчанию при добавлении стартера c Edgware

    spring-cloud release train application.yml: server.port: 8761 spring.application.name: registry eureka: client.register-with-eureka= false server.enable-self-preservation= false
  115. Кто это у нас там спрятался? Смотрим какие сервисы загрузились

    $ http :8082/health { "description": "Remote status from Eureka server", "status": "DOWN" } { "description": "Spring Cloud Eureka Discovery Client", "status": "UP" } Spring Boot 1.x.x
  116. Кто это у нас там спрятался? Выключаем security для управляющий

    API management: security: enabled: false Теперь $ http :8082/health Spring Boot 1.x.x
  117. { "description": "Spring Cloud Eureka Discovery Client", "discoveryComposite": { "description":

    "Spring Cloud Eureka Discovery Client", "discoveryClient": { "description": "Spring Cloud Eureka Discovery Client", "services": [ "examinator-service", "math-service", "theology-service" ], "status": "UP" }, "eureka": { "applications": { "EXAMINATOR-SERVICE": 1, "MATH-SERVICE": 1, "THEOLOGY-SERVICE": 1 }, "description": "Remote status from Eureka server", "status": "UP" }, "status": "UP" }, "diskSpace": { "free": 190254260224, "status": "UP", "threshold": 10485760, "total": 499055067136 }, ... Spring Boot 1.x.x
  118. Кто это у нас там спрятался? Смотрим какие сервисы загрузились

    $ http :8082/actuator/health { "status": "UP" } $ http :8082/actuator/metrics ... $ http :8082/actuator ... Spring Boot 2.x.x
  119. True Discovery • Изменить логику резолва сервиса использую имя сервиса

    • Написать контроллер, который возвращает все зарегистрированные сервисы • Запустить несколько сервисов одного типа, посмотреть результат Use @LoadBalanced or LoadBalancerInterceptor Luke @Autowired private DiscoveryClient discoveryClient; discoveryClient.getServices() discoveryClient.getInstances("theology")
  120. 252 Service Client Registry-aware HTTP Client Service Registry Service Instance

    1 Service Instance N Service Instance ... Load balance request Client side discovery Ribbon + Eureka Discovery client
  121. 253 Service Client Service Registry Service Instance 1 Service Instance

    N Service Instance ... Load balance request Router/Proxy Server side discovery
  122. Принципы SOA 256 1. Standardized service contract 2. Loose coupling

    3. Encapsulation 4. Reusability 5. Autonomy 6. Statelessness 7. Discoverability
  123. Принципы SOA 257 1. Standardized service contract 2. Loose coupling

    3. Encapsulation 4. Reusability 5. Autonomy 6. Statelessness 7. Discoverability
  124. Fluent annotations 258 @Getter // generate getters @Setter // generate

    setters @Aspect // we are an aspect @ToString // generate toString() @EnableWs // SOAP is so enterprisy, we definitely need it @Endpoint // Seriously, just read above @EnableWebMvc // we want MVC @EnableCaching // and we want to cache stuff @Configuration // this class can configure itself @RestController // we want some REST @XmlRootElement // this component is marshallable @EnableWebSocket // we want web socket, it's so new-generation @RedisHash("cat") // this class is an entity saved in redis @EnableScheduling // we want scheduled tasks @EnableWebSecurity // and some built-in security @NoArgsConstructor // generate no args constructor @ContextConfiguration // we want context configuration for unit testing @SpringBootApplication // this is a Sprint Boot application @Accessors(chain = true) // getters/setters are chained (ala jQuery) @EnableAspectJAutoProxy // we want AspectJ auto proxy @EnableAutoConfiguration // and auto configuration
  125. Парадокс централизации Чтобы эффективно разрабатывать распределённые приложения, нам нужны очень

    хорошие централизованные библиотеки и инструменты Например: логирование, health-checking, метрики, обработка типовых ошибок, автодокументирование 259
  126. Парадокс централизации Чтобы эффективно разрабатывать распределённые приложения, нам нужны очень

    хорошие централизованные библиотеки и инструменты Но: не выносите бизнес-логику или доменные объекты! Не размывайте бизнес-контекст вашего API 260
  127. Your own Config Server • Создать модуль config-server compile 'org.springframework.cloud:spring-cloud-starter-config'

    compile 'org.springframework.cloud:spring-cloud-config-server' @EnableConfigServer spring: cloud: config: server: git: uri: https://github.com/lavcraft/spring-boot-and-cloud-workshop-joker2017-configs.git
  128. Проблема курица и яйца Откуда брать координаты eureka? • из

    сonfig-server (для всех же одинаковые) • из локальных настроек (ENV/*.yml/etc) Откуда брать координаты сonfig-server • из eureka • из локальных настроек (ENV/*.yml/etc)
  129. Config Server Discovery Через Service Discovery App Instance #0 App

    Instance #N Config Server По прямой ссылке Eureka
  130. Config Server Discovery Через Service Discovery App Instance #0 App

    Instance #N Eureka Config Server register on start get Config Server by name
  131. Config Server Discovery Через Service Discovery App Instance #0 App

    Instance #N Eureka Config Server register on start return config server instances location get Config Server by name
  132. Config Server Discovery Через Service Discovery App Instance #0 App

    Instance #N Eureka Config Server register on start return config server instances location get Config Server by name get configs for App
  133. Config Server Discovery По прямой ссылке App Instance #0 App

    Instance #N Config Server get configs for App Http Load Balancer
  134. Начнём с примера @RestController @RefreshScope public class TestController { @Value("${foo.bar}")

    String test; @GetMapping("/foo") public String getFooBar() { return test; } }
  135. Начнём с примера @RestController @RefreshScope public class TestController { @Value("${foo.bar}")

    String test; @GetMapping("/foo") public String getFooBar() { return test; } }
  136. Начнём с примера @RestController @RefreshScope public class TestController { @Value("${foo.bar}")

    String test; @GetMapping("/foo") public String getFooBar() { return test; } }
  137. Три простых шага 1. Ставим @RefreshScope над бином который нужно

    обновлять 2. Обновляем в config server foo.bar: some val a. Можно не править а дёрнуть /env если есть actuator 3. Дёргаем у приложения /refresh a. Для этого нужен actuator
  138. Spring Cloud Bus dependencyManagement { imports { mavenBom 'org.springframework.cloud:spring-cloud-bus:2.1.1.RELEASE' }

    } dependencies { compile 'org.springframework.cloud:spring-cloud-starter-bus-amqp' }
  139. Теперь достаточно /refresh на config-server 1. Все приложения обновят конфигу

    при получении refresh ивента в rabbitmq 2. Для автоматизации можно использовать config-monitor + git webhook compile 'org.springframework.cloud:spring-cloud-config-monitor' 3. Не забывать @RefreshScope
  140. spring-cloud-config-monitor Config Server Экзаминатор Экзаминатор GitHub commit Config Monitor webhook

    Queue refresh event Теология Теология Теология ... auto /refresh on event
  141. Зависимости Spring Cloud Gateway dependencies { compile 'org.springframework.cloud:spring-cloud-starter-gateway' //for webflux

    compile 'org.springframework.cloud:spring-cloud-gateway-webflux' //or for mvc compile 'org.springframework.cloud:spring-cloud-gateway-mvc' } Spring boot 2.x.x
  142. Зависимости Spring Cloud Gateway dependencies { compile 'org.springframework.cloud:spring-cloud-starter-gateway' //for webflux

    compile 'org.springframework.cloud:spring-cloud-gateway-webflux' //or for mvc compile 'org.springframework.cloud:spring-cloud-gateway-mvc' } application.yml: spring: cloud.gateway.discovery.locator.enabled: true Spring boot 2.x.x
  143. 298

  144. Gateway + Hystrix + Dashboard + Discovery dependencies { compile

    'org.springframework.boot:spring-boot-starter-web' compile 'org.springframework.boot:spring-boot-starter-actuator' compile 'org.springframework.cloud:spring-cloud-starter-zuul' compile 'org.springframework.cloud:spring-cloud-starter-eureka' compile 'org.springframework.cloud:spring-cloud-starter-hystrix' compile 'org.springframework.cloud:spring-cloud-starter-hystrix-dashboard' } Spring boot 1.x.x
  145. 318 Rent Service Payment Service No TraceId No SpanId TraceId

    = X SpanId = A TraceId = X SpanId = B TraceId = X SpanId = C
  146. 319 Rent Service Payment Service Blockchain Service No TraceId No

    SpanId TraceId = X SpanId = A TraceId = X SpanId = B TraceId = X SpanId = C TraceId = X SpanId = D TraceId = X SpanId = D TraceId = X SpanId = F
  147. 320 Rent Service Payment Service Security Service Blockchain Service No

    TraceId No SpanId TraceId = X SpanId = A TraceId = X SpanId = B TraceId = X SpanId = C TraceId = X SpanId = D TraceId = X SpanId = D TraceId = X SpanId = E TraceId = X SpanId = E TraceId = X SpanId = F TraceId = X SpanId = G
  148. 321 Rent Service Payment Service Security Service Blockchain Service No

    TraceId No SpanId TraceId = X SpanId = A TraceId = X SpanId = B TraceId = X SpanId = B TraceId = X SpanId = C TraceId = X SpanId = C TraceId = X SpanId = D TraceId = X SpanId = D TraceId = X SpanId = E TraceId = X SpanId = E TraceId = X SpanId = F TraceId = X SpanId = G
  149. 322 Rent Service Payment Service Security Service Blockchain Service TraceId

    = X SpanId = A No TraceId No SpanId TraceId = X SpanId = A TraceId = X SpanId = A TraceId = X SpanId = B TraceId = X SpanId = B TraceId = X SpanId = C TraceId = X SpanId = C TraceId = X SpanId = D TraceId = X SpanId = D TraceId = X SpanId = E TraceId = X SpanId = E TraceId = X SpanId = F TraceId = X SpanId = G
  150. 323

  151. 324

  152. 327 Rent Service Payment Service SpanId = B Client Send

    SpanId = B Server Received TraceId = X SpanId = A TraceId = X SpanId = C
  153. 328 Rent Service Payment Service SpanId = B Client Send

    SpanId = B Server Received SpanId = B Server Send TraceId = X SpanId = A TraceId = X SpanId = C
  154. 329 Rent Service Payment Service SpanId = B Client Send

    SpanId = B Server Received SpanId = B Client Received SpanId = B Server Send TraceId = X SpanId = A TraceId = X SpanId = C
  155. Not smart = This is main documentation This document describes

    how to be the most fundamental and important document in the world of documents ... COPY-PASTE documentation from another document ... 331
  156. Not so smart = This is main documentation This document

    describes how to be the most fundamental and important document in the world of documents include::https://raw.github.com/asciidoctor/asciidoctor/master/Gemfile[] include::../other.adoc[] include::/home/tolkv/git/docs-0/superdoc.adoc[] 332
  157. Really smart = This is main documentation This document describes

    how to be the most fundamental and important document in the world of documents include::https://raw.github.com/asciidoctor/asciidoctor/master/Gemfile[] include::gradle://gradle-advanced:service-with-deps:1.0/deps.adoc[] include::gradle://:service/doc.adoc[] 333
  158. Payment Service[jar,doc] Insurance Service [jar,doc] One Point of View [UberDoc.zip]

    Rent Service[jar,doc] Other Service [jar,doc] Агрегация информации 334
  159. Парадокс централизации Чтобы эффективно разрабатывать распределённые приложения, нам нужны очень

    хорошие централизованные библиотеки и инструменты Например: логирование, health-checking, метрики, обработка типовых ошибок, автодокументирование 335
  160. Парадокс централизации Чтобы эффективно разрабатывать распределённые приложения, нам нужны очень

    хорошие централизованные библиотеки и инструменты Но: не выносите бизнес-логику или доменные объекты! Не размывайте бизнес-контекст вашего API 336