/microservices-practical-activity-docker

Spring Boot, Spring Data JPA, Spring Cloud , Docker ....

Primary LanguageJava

Les bonnes pratiques et challenges des architectures distribuées basées sur les Micro-services

Remember:

“IOC”, “DI” mais c’est quoi au juste?

L’inversion de contrôle ou IOC pour inversion of control est un principe général de conception dont le but est de diminuer le couplage entre les composants logiciels.

L’injection de dépendances ou DI pour “dependency injection”, est-elle une méthode permettant d’appliquer le principe d’IOC. Le principe est tout simple, il consiste à passer (injecter) un composant logiciel (une classe par exemple) à un autre composant logiciel qui l’utilise, permettant ainsi d’éviter la dépendance d’un composant à un autre et ainsi améliorer la souplesse de l’application.

Content

  1. Basics
    1. Architecture Monolotique et contraintes
    2. Architecture Micro-services et contraintes
    3. Gateway
    4. Modèle de communication entre les microservices
    5. Security challenge
    6. How to deal with my data ?
    7. How to deal with my data : CQRS & Event Sourcing
  2. Demo
    1. Use case
    2. Customer service
      1. Dependencies
      2. Properties file
      3. JPA Entity
      4. JPA Repository
      5. Service layer
      6. DTOs
      7. Mapping
      8. Rest Controller
      9. Customer Service tests
      10. Documenting our service with openAPI (Swagger)
    3. Billing Service
      1. Billing Service Dependencies
      2. Billing Service Properties file
      3. Billing Service Entity
      4. Billing Service OpenFeign Client
      5. Billing Service DTOs
      6. Billing Service Mapping
      7. Billing Service Service layer
      8. Billing Service Rest Controller
      9. Billing Service documentation
      10. Billing Service tests
    4. Discovery service
    5. Gateway service
    6. [Refactoring]
      1. Handling Exceptions
  3. Dockerizing our micro-services
    1. Dockerise Using Docker File and Docker compose
    2. Dockerise Using Kubernetes

Architecture Monolotique et contraintes:

  • Basé sur plusieurs modèle qui ils sont intégrés dans la même application.
  • Une seule base de données.
  • Une seule technologie (Côut de formation).
  • Processus Unique (Si une fonctionnalité arrête, l'application arrête completement).
  • Difficiles à maintenir.
  • Difficile à tester.
  • Mise en production prend beaucoup de temps.
  • Redéploiement à froid (obliger d'acrrêter l'app et la redéploier).
  • Performances (Scalabilité).

Architecture Micro-services et contraintes:

  • Un ensembles des modèles séparés avec des bases de données séparées.
  • Un ensembles des petites applications.
  • Les application front-end ne doivent pas forcement savoir les microservices, ils ont besoin juste de connecter avec la Gateway.
  • Gateway pour chaque requete reçus il va se charger d'acheminer la requete vers le bon Micro-service (Routage).
  • Pour svoir dans quelle machine se trouve le Microservice, Gateway utilise Discovery service, pour un mappage dynamique surtout si on démarre plusieurs instance d'un microservice.
  • Discovery service se chargé de publier ou d'enregistrer toutes les instance des Micro-services disponibles.
  • Chaque microservice qu'il démarre va enregistrer son adresse IP, nom, et port dans le discovery service.
  • Pour ne pas avoir des conflits de configuration, on utilise le Config service qui va se charger de cnetraliser la configuration globale en commun.
  • Le microservice reçoit sa configuration à chaud c à d sans avoir besoin l'arrêter.

Microservice cherche sa configuration dans Config serviceMicroservice DémarreMicroservice DémarreMicroservice Démarre → .... youtba3

vs

  • Event Bus: pour une communication asynchronne entre les microservices.

Avantages

  • Performance (Scalabilité Horzontale) : Si on a un problème de monté en charge il suffit démarrer d'autres instances du micro-service affecter, il va s'enregistrer dans le discovery service et il va etre disponible, la Gateway dans ce cas là va utiliser un système d'équilibrage de charge load balancer entre les instances.
  • Processus séparés.
  • Faciles à développer à tester et à maintenir.
  • Mise en production rapide des micro-services.
  • Redéploiement à chaud.
  • Téchnologies déffirentes.
  • Equipes indépendentes.
  • Facile à appliquer l'agilité.
  • Facile à mettre en oeuvre TDD (Test puis fonctionnalité).

Gateway

ZUUL Spring Cloud Gateway
Modèle Multi thread avec des entrés sorties bloquantes Modèle Single thread avec des entrés sorties non bloquantes
Thread pour chaque requête Un thread pour toutes les requetes (Event loop)
Thread peut etre mobiliser pour une longue durée Asychronisation, buffering pour servir plusieurs reqêtes
N'est pas scalable virticalement Scalable virticalement
Limité par le nombre maximal des threads limité par les ressources uniquement

Modèle de communication entre les microservices

Modèle synchrone Modèle asynchrone
REST avec Open Feign Par des Event Bus/ Broker avec Kafka, RabbitMQ
Utiliser l'API du service pour envoyer des requêtes HTTP Les service fait un subscribe dans event bus et attends pour les informations arrivées
1 1

Security challenge

Modèle d'authentification Statful Modèle d'authentification Statless
Les informations de la session s'enregistrent dans le côté serveur, le client reçoit un Session ID unique à conserver dans les Cookies pour l'envoyer avec les prochaines requêtes Les données de la session sont enregistrées dans un jeton/ token d'authentification délivré au client
Utilisé dans la majorité des cas dans les applications monolitiques
- Ex. JWT..
- Plus adapté au Micro-services car on a pas de session partagé entre les services

Quelques outils à savoir en ralation avec sécurité:

How to deal with my data ?

  • Garder l'architecture Microservices mais avec une seule base de données ??
  • L'ideal c'est d'utiliser une DB pour chaque Micro-service, pour garder la performance et la scalabilité.
  • Plus Ideal : Utiliser les deux au même temps avec un Event Bus: c'est d'avoir des microservices avec leurs bases de données + un qui a une bas de données globale, dans laquel il stock tous les données, Ce microservice attend les evenements du Broker (Event Bus) et mis à jour la base globale.

3

  • On peut fair la même chose si par exemple on a besoin à une base de données pour un moteur de recherche (Elasticsearch), il suffit de créer un autre service handler.

How to deal with my data : CQRS & Event Sourcing

  • CQRS: Command Query Responsability Segregation
  • A pattern that separates read and update operations for a data store. Implementing CQRS in your application can maximize its performance, scalability, and security.
  • Séparer le bus de lecture du bus d'écriture.
  • Le bus de Command permet de modifier les données
  • Le bus de Query permet de la lecture des données.
  • Event Sourcing: An patern qui consist à ne pas enregister le dernier état de l'application dans la base de donnée mais, d'enregister tous les élements (evenements) de l'application, et qui permet de retrouver l'état de notre app (Je veux l'état de ma base de données d'un mois avant...).
  • Outils : Spring cloud, AXION Framework.

Demo

Use case :

  • Respecter tous les bonnes pratiques:

4

5

Travail à faire :
 Suivre les vidéos Bonnes pratiques de Architectures micro-services :
 1. Vidéo 1 : Concepts de bases
 2.  Vidéo 2 à 5 : Mise en oeuvre d'une application distribuée basée sur deux micro-services en utilisant les bonnes pratiques  :
  - Couches DA0, Service, Web, DTO
  - Utilisation de MapStruct pour le mapping entre les objet Entities et DTO
  - Génération des API-DOCS en utilisant SWAGGER3 (Open API)
  - Communication entre micro-services en utilisant OpenFeign
  - Spring Cloud Gateway
  - Eureka Discovery Service
Travail à rendre Dimanche 31 Octobre
  - etc...

Customer Service

DTO: Data transfer object => Objects adapted to the UI layer, it is just a fransfer of the data.

Customer Service Dependencies

in build.gradle file (or pom.xml) if we use maven we add dependencies.

dependencies {
	implementation 'org.springframework.boot:spring-boot-starter-data-jpa'
	implementation 'org.springframework.boot:spring-boot-starter-validation'
	implementation 'org.springframework.boot:spring-boot-starter-webflux'
	implementation 'org.springframework.cloud:spring-cloud-starter-gateway'
	implementation 'org.springframework.cloud:spring-cloud-starter-netflix-eureka-client'
	compileOnly 'org.projectlombok:lombok'
	runtimeOnly 'com.h2database:h2'
	annotationProcessor 'org.projectlombok:lombok'
	testImplementation 'org.springframework.boot:spring-boot-starter-test'
	testImplementation 'io.projectreactor:reactor-test'
	implementation 'org.mapstruct:mapstruct:1.5.3.Final'
	annotationProcessor 'org.mapstruct:mapstruct-processor:1.5.3.Final'
	// https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-ui
	implementation 'org.springdoc:springdoc-openapi-ui:1.6.12'
}

If you are a Maven user, add the following dependencies to your pom.xml file:

<dependencies>
    <!-- ... -->
    <dependency>
        <groupId>org.mapstruct</groupId>
        <artifactId>mapstruct</artifactId>
        <version>1.4.2.Final</version>
    </dependency>   
</dependencies>

<!-- ... -->
<build>
    <plugins>
        <!-- ... -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <version>3.8.1</version>
            <configuration>
                <source>17</source> <!-- depending on your project. In this example, Java 11 is used -->
                <target>17</target> <!-- depending on your project. In this example, Java 11 is used -->
                <annotationProcessorPaths>
                    <!-- Here we can add the path for lombok -->
                    <path>
                        <groupId>org.projectlombok</groupId> 
                        <artifactId>lombok</artifactId>
                        <version>1.18.16</version>
                    </path>
                    <path>
                        <groupId>org.mapstruct</groupId> 
                        <artifactId>mapstruct-processor</artifactId>
                        <version>1.4.2.Final</version>
                    </path>
                </annotationProcessorPaths>
            </configuration>
        </plugin>
    </plugins>
</build>

from: https://maven.apache.org/plugins/maven-compiler-plugin/compile-mojo.html

Classpath elements to supply as annotation processor path. If specified, the compiler will detect annotation processors only in those classpath elements. If omitted, the default classpath is used to detect annotation processors. The detection itself depends on the configuration of annotationProcessors.

Each classpath element is specified using their Maven coordinates (groupId, artifactId, version, classifier, type). Transitive dependencies are added automatically. Example:
<configuration>
  <annotationProcessorPaths>
    <path>
      <groupId>org.sample</groupId>
      <artifactId>sample-annotation-processor</artifactId>
      <version>1.2.3</version>
    </path>
    <!-- ... more ... -->
  </annotationProcessorPaths>
</configuration>

Customer Service Properties file

server.port=8082
spring.application.name=CUSTOMER-SERVER
spring.h2.console.enabled=false
spring.cloud.discovery.enabled=false # prevent server to be registred to discovery service
spring.datasource.url=jdbc:h2:mem:customer-db

Customer Service Entity

@Data
@Entity
@NoArgsConstructor @Builder
@AllArgsConstructor
public class Customer {
    @Id
    private String id;
    @NotBlank
    private String name;
    @Email
    private String email;
}

Customer Service Repository

@Repository
public interface CustomerRepository extends JpaRepository<Customer, String> {
}

Customer Service Service layer

@Service
@Transactional
public interface CustomerService {

    CustomerResponseDTO saveCustomer(CustomerRequestDTO customerRequestDTO);

    CustomerResponseDTO getCustomer(String id);

    CustomerResponseDTO updateCustomer(String id, CustomerRequestDTO customerRequestDTO);

    boolean deleteCustomer(String id);

    List<CustomerResponseDTO> getAllCustomers(int page, int size);
}

Implementation

@Service
@Transactional
@AllArgsConstructor
public class CustomerServiceImpl implements CustomerService {

    private CustomerRepository customerRepository;
    private CustomerMapper customerMapper;

    @Override
    public CustomerResponseDTO saveCustomer(CustomerRequestDTO customerRequestDTO){
        Customer customer = customerMapper.toCustomer(customerRequestDTO);
        customer.setId(UUID.randomUUID().toString());
        Customer savedCustomer = customerRepository.save(customer);
        return customerMapper.toCustomerResponse(savedCustomer);
    }

    @Override
    public CustomerResponseDTO getCustomer(String id){
        Customer customer = customerRepository
                .findById(id)
                .orElseThrow( () -> {
                    return  new  RuntimeException(String.format("Customer with ID: %s Not found !", id));
                });
        return customerMapper.toCustomerResponse(customer);
    }

    @Override
    public CustomerResponseDTO updateCustomer(String id, CustomerRequestDTO customerRequestDTO){
        Customer customer = customerRepository
                .findById(id)
                .orElseThrow( () -> {
                    return  new  RuntimeException(String.format("Customer with ID: %s Not found !", id));
                });

        Customer customerMap = customerMapper.toCustomer(customerRequestDTO);
        customerMap.setId(id);
        Customer savedCustomer = customerRepository.save(customerMap);
        return customerMapper.toCustomerResponse(savedCustomer);
    }

    @Override
    public boolean deleteCustomer(String id){
        customerRepository.deleteById(id);
        return true;
    }

    @Override
    public List<CustomerResponseDTO> getAllCustomers(int page, int size){
        Page<Customer> customerPage = customerRepository.findAll(PageRequest.of(page, size));
        List<CustomerResponseDTO> customerResponseDTOList =
                customerPage.getContent().stream().map(customer -> customerMapper.toCustomerResponse(customer)).collect(Collectors.toList());
        return customerResponseDTOList;
    }

}

Customer Service DTOs

@Data
@AllArgsConstructor
@NoArgsConstructor
@Builder

public class CustomerRequestDTO {
    private String id;
    private String name;
    private String email;
}
@Data @AllArgsConstructor @NoArgsConstructor @Builder
public class CustomerResponseDTO {
    private String id;
    private String name;
    private String email;
}

Customer Service Mapping

Documentation: https://mapstruct.org/documentation/stable/reference/html/ Get Statted with mapstruct : https://mapstruct.org/

CustomerMapper interface in the mappers package

@Mapper(componentModel = "spring")
public interface CustomerMapper {
    CustomerResponseDTO toCustomerResponse(Customer customer);
    Customer toCustomer(CustomerRequestDTO customerRequestDTO);
}

Customer Service Rest Controller

@RestController
@RequestMapping(path = "/api-v1/customer-service")
@AllArgsConstructor
public class CustomerController {
    private CustomerService customerService;

    @GetMapping("/")
    public List<CustomerResponseDTO> getCustomerList(@RequestParam(name = "page", defaultValue = "0") int page,
                                                     @RequestParam(name = "size", defaultValue = "10") int size ){

        return customerService.getAllCustomers(page, size);
    }

    @GetMapping ("/{customerId}")
    public CustomerResponseDTO getCustomer(@PathVariable(name = "customerId") String id){
        return customerService.getCustomer(id);
    }

    @PostMapping("/")
    public CustomerResponseDTO insertCustomer(@RequestBody CustomerRequestDTO customerRequestDTO){
        return customerService.saveCustomer(customerRequestDTO);
    }

    @PutMapping("/{customerId}")
    public CustomerResponseDTO updateCustomer(@PathVariable(name = "customerId") String id,
                                              @RequestBody CustomerRequestDTO customerRequestDTO){
        return customerService.updateCustomer(id, customerRequestDTO);
    }

    @DeleteMapping("/{customerId}")
    public boolean updateCustomer(@PathVariable(name = "customerId") String id){
        return customerService.deleteCustomer(id);
    }

}

Customer Service tests

Adding 2 Customers at the begining

@Bean
	CommandLineRunner start(CustomerRepository customerRepository){
		return args -> {
			// Create just 2 customers

			List.of("Ahmed", "Essadeq").forEach( c ->{
				Customer customer = new Customer(null, c, String.format("%s_email@gmail.com", c));
				customerRepository.save(customer);
			});
		};
	}

H2-console consulting:

Problem:

  • Do not have access to http://localhost:8082/h2-console
Whitelabel Error Page
This application has no configured error view, so you are seeing this as a fallback.

Wed Oct 26 15:39:39 WEST 2022
[bb486a63-3] There was an unexpected error (type=Not Found, status=404).
org.springframework.web.server.ResponseStatusException: 404 NOT_FOUND
	at org.springframework.web.reactive.resource.ResourceWebHandler.lambda$handle$1(ResourceWebHandler.java:408)
	Suppressed: The stacktrace has been enhanced by Reactor, refer to additional information below: 
Error has been observed at the following site(s):
	*__checkpoint ⇢ org.springframework.cloud.gateway.filter.WeightCalculatorWebFilter [DefaultWebFilterChain]
	*__checkpoint ⇢ HTTP GET "/h2-console" [ExceptionHandlingWebHandler]
Original Stack Trace: ...
  • Tried to install devtools, but did not resolve the problem.
  • spring.h2.console.enabled=true did not resolve the problem.

But the application works fine:

Visiting http://localhost:8082/api-v1/customers/

[{"id":"96fe8548-ceda-420c-8a50-b4d4ab3b75c1","name":"Ahmed","email":"Ahmed_email@gmail.com"},{"id":"d62da59d-6483-4535-948b-63fd2751e81d","name":"Essadeq","email":"Essadeq_email@gmail.com"}]

Visiting : http://localhost:8082/api-v1/customers/96fe8548-ceda-420c-8a50-b4d4ab3b75c1

{"id":"96fe8548-ceda-420c-8a50-b4d4ab3b75c1","name":"Ahmed","email":"Ahmed_email@gmail.com"}

Tested with restClient VScode extension : Usage : https://marketplace.visualstudio.com/items?itemName=MadsKristensen.RestClient Usage 2: https://medium.com/refinitiv-developer-community/how-to-test-rest-api-with-visual-studio-code-rest-client-extensions-9f2e061d0299

File: FILE_LINK

@hostname = localhost
@port = 8082
@contentType = application/json
@baseUrl = http://{{hostname}}:{{port}}/api-v1/customers/
POST {{baseUrl}} HTTP/1.1
Content-Type: {{contentType}}

{
    "name":"Laila",
    "email":"laila@gmail.com"
}

# edit
PUT {{baseUrl}}e07ba26b-6620-4733-a9d0-47cc9b2c2f59 HTTP/1.1
Content-Type: {{contentType}}
{
    "id" : null,
    "name":"Laila_edited",
    "email":"laila@gmail.com"
}

# list all 

GET {{baseUrl}} HTTP/1.1

Tested also with Postman ==> everything works fine [ 😄 DONE]

Documenting our service with openAPI (Swagger)

Adding the dependency

// https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-ui
implementation 'org.springdoc:springdoc-openapi-ui:1.6.12'

Visiting : http://localhost:8082/v3/api-docs we will have our API documentation, which can be used with any testion tool (Postman ...).

INFO: https://www.baeldung.com/spring-rest-openapi-documentation

To use a custom path, we can indicate in the application.properties file:

springdoc.api-docs.path=/api-docs

Now we'll be able to access the docs at:

http://localhost:8080/api-docs/

The OpenAPI definitions are in JSON format by default. For yaml format, we can obtain the definitions at:

http://localhost:8080/api-docs.yaml

Besides generating the OpenAPI 3 specification itself, we can integrate springdoc-openapi with Swagger UI so that we can interact with our API specification and exercise the endpoints.

The springdoc-openapi dependency already includes Swagger UI, so we're all set here.

We can simply access the API documentation at:

http://localhost:8080/swagger-ui.html

  • OpenAPI is the equivalent of WSDL for SOAP based services
  • What's that ?
REST versus SOAP. It’s been an issue for a while now. And really, they’re just two answers to the 
same question: how to access web services.

But deciding one over the other can be surprisingly difficult.

SOAP (Simple Object Access Protocol) is a standards-based web services access protocol that has 
been around for a long time. Originally developed by Microsoft, SOAP isn’t as simple as the 
acronym would suggest.

REST (Representational State Transfer) is another standard, made in response to SOAP’s 
shortcomings. It seeks to fix the problems with SOAP and provide a simpler method of accessing web 
services. 

What about GraphQL?
Of course, GraphQL has recently made a huge splash, which we’ve spoken of at length in other 
articles. But it’s still not as standardized as REST and SOAP, so in this article we’re just going 
to focus on those two.

Problems 404 Error on http://localhost:8082/h2-console and http://localhost:8082/v3/api-docs

Whitelabel Error Page
This application has no configured error view, so you are seeing this as a fallback.

Wed Oct 26 17:03:21 WEST 2022
[75afbc90-6] There was an unexpected error (type=Not Found, status=404).

Raison:

As per your logs, I found that, you are using an embedded server, other than tomcat, i.e, spring-boot-starter-reactor-netty that comes along with spring webflux dependency.

H2ConsoleAutoConfiguration will not be executed for spring webflux & netty(reactor based), as H2 console will only be available to servlet based applications. So, you have to configure H2 server manually in this spring boot application, with spring webflux & netty.

Suggested solutions:
https://stackoverflow.com/questions/63646864/spring-boot-h2-console-returns-404

https://stackoverflow.com/questions/52949088/h2-db-not-accessible-at-localhost8080-h2-console-when-using-webflux

Solution for : http://localhost:8082/v3/api-docs

  • Adding the dependency to duild.gradle :
// https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-webflux-ui
implementation 'org.springdoc:springdoc-openapi-webflux-ui:1.6.12'

Instead of

// https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-ui
implementation 'org.springdoc:springdoc-openapi-ui:1.6.12'
  • Adding the config in the Application
@SpringBootApplication
@OpenAPIDefinition(info = @Info(title = "APIs", version = "1.0", description = "Documentation APIs v1.0"))
public class CustomerServiceApplication {

	public static void main(String[] args) {
		SpringApplication.run(CustomerServiceApplication.class, args);
	}

	@Bean
	CommandLineRunner start(CustomerRepository customerRepository){
		return args -> {
			// Create just 2 customers

			List.of("Ahmed", "Essadeq").forEach( c ->{
				Customer customer = new Customer(UUID.randomUUID().toString(), c, String.format("%s_email@gmail.com", c));
				customerRepository.save(customer);
			});
		};
	}
}

// here is the description of our API

{
   "openapi":"3.0.1",
   "info":{
      "title":"APIs",
      "description":"Documentation APIs v1.0",
      "version":"1.0"
   },
   "servers":[
      {
         "url":"http://localhost:8082",
         "description":"Generated server url"
      }
   ],
   "paths":{
      "/api-v1/customers/{customerId}":{
         "get":{
            "tags":[
               "customer-controller"
            ],
            "operationId":"getCustomer",
            "parameters":[
               {
                  "name":"customerId",
                  "in":"path",
                  "required":true,
                  "schema":{
                     "type":"string"
                  }
               }
            ],
            "responses":{
               "200":{
                  "description":"OK",
                  "content":{
                     "*/*":{
                        "schema":{
                           "$ref":"#/components/schemas/CustomerResponseDTO"
                        }
                     }
                  }
               }
            }
         },
         "put":{
            "tags":[
               "customer-controller"
            ],
            "operationId":"updateCustomer",
            "parameters":[
               {
                  "name":"customerId",
                  "in":"path",
                  "required":true,
                  "schema":{
                     "type":"string"
                  }
               }
            ],
            "requestBody":{
               "content":{
                  "application/json":{
                     "schema":{
                        "$ref":"#/components/schemas/CustomerRequestDTO"
                     }
                  }
               },
               "required":true
            },
            "responses":{
               "200":{
                  "description":"OK",
                  "content":{
                     "*/*":{
                        "schema":{
                           "$ref":"#/components/schemas/CustomerResponseDTO"
                        }
                     }
                  }
               }
            }
         },
         "delete":{
            "tags":[
               "customer-controller"
            ],
            "operationId":"updateCustomer_1",
            "parameters":[
               {
                  "name":"customerId",
                  "in":"path",
                  "required":true,
                  "schema":{
                     "type":"string"
                  }
               }
            ],
            "responses":{
               "200":{
                  "description":"OK",
                  "content":{
                     "*/*":{
                        "schema":{
                           "type":"boolean"
                        }
                     }
                  }
               }
            }
         }
      },
      "/api-v1/customers/":{
         "get":{
            "tags":[
               "customer-controller"
            ],
            "operationId":"getCustomerList",
            "parameters":[
               {
                  "name":"page",
                  "in":"query",
                  "required":false,
                  "schema":{
                     "type":"integer",
                     "format":"int32",
                     "default":0
                  }
               },
               {
                  "name":"size",
                  "in":"query",
                  "required":false,
                  "schema":{
                     "type":"integer",
                     "format":"int32",
                     "default":10
                  }
               }
            ],
            "responses":{
               "200":{
                  "description":"OK",
                  "content":{
                     "*/*":{
                        "schema":{
                           "type":"array",
                           "items":{
                              "$ref":"#/components/schemas/CustomerResponseDTO"
                           }
                        }
                     }
                  }
               }
            }
         },
         "post":{
            "tags":[
               "customer-controller"
            ],
            "operationId":"insertCustomer",
            "requestBody":{
               "content":{
                  "application/json":{
                     "schema":{
                        "$ref":"#/components/schemas/CustomerRequestDTO"
                     }
                  }
               },
               "required":true
            },
            "responses":{
               "200":{
                  "description":"OK",
                  "content":{
                     "*/*":{
                        "schema":{
                           "$ref":"#/components/schemas/CustomerResponseDTO"
                        }
                     }
                  }
               }
            }
         }
      }
   },
   "components":{
      "schemas":{
         "CustomerRequestDTO":{
            "type":"object",
            "properties":{
               "id":{
                  "type":"string"
               },
               "name":{
                  "type":"string"
               },
               "email":{
                  "type":"string"
               }
            }
         },
         "CustomerResponseDTO":{
            "type":"object",
            "properties":{
               "id":{
                  "type":"string"
               },
               "name":{
                  "type":"string"
               },
               "email":{
                  "type":"string"
               }
            }
         }
      }
   }
}

Visiting : http://localhost:8082/webjars/swagger-ui/index.html

6

⚠️

By careful

- By mistake, I was adding the gateway dependency to my app, so the dependency manager keeped 
- install Spring webflux for me instead of Spring Web.

Billing Service

Billing Service Dependencies

Here is the dependencies and the configuration

[Show all code]

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.5</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>me.elaamiri</groupId>
    <artifactId>my-billing-service</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>my-billing-service</name>
    <description>my-billing-service</description>
    <properties>
        <java.version>17</java.version>
        <spring-cloud.version>2021.0.4</spring-cloud.version>
        <org.mapstruct.version>1.5.3.Final</org.mapstruct.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-validation</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>

        <dependency>
            <groupId>com.h2database</groupId>
            <artifactId>h2</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct</artifactId>
            <version>${org.mapstruct.version}</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-ui -->
        <dependency>
            <groupId>org.springdoc</groupId>
            <artifactId>springdoc-openapi-ui</artifactId>
            <version>1.6.12</version>
        </dependency>
    </dependencies>
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring-cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>


        </dependencies>
    </dependencyManagement>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <excludes>
                        <exclude>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                        </exclude>
                    </excludes>
                </configuration>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.8.1</version>
                <configuration>
                    <source>${java.version}</source> <!-- depending on your project -->
                    <target>${java.version}</target> <!-- depending on your project -->
                    <annotationProcessorPaths>
                        <path>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                            <version>1.18.16</version>
                        </path>
                        <path>
                            <groupId>org.mapstruct</groupId>
                            <artifactId>mapstruct-processor</artifactId>
                            <version>${org.mapstruct.version}</version>
                        </path>
                        <!-- other annotation processors -->
                    </annotationProcessorPaths>
                </configuration>
            </plugin>
        </plugins>
    </build>

</project>

Billing Service Properties file

server.port=8083
spring.application.name=BILLING-SERVER
spring.h2.console.enabled=true
spring.cloud.discovery.enabled=false
spring.datasource.url=jdbc:h2:mem:billing-db

Billing Service Entity

Here is our Invoice entity

// facture
@Data @NoArgsConstructor @AllArgsConstructor @Builder
@Entity
public class Invoice {
    @Id
    private String id;
    @Temporal(TemporalType.DATE)
    private Date date;
    private BigDecimal amount;
    @NotBlank
    private String customerID;
    @Transient // Not to be persistent
    private Customer customer;
}

And the helperModel Customer

@Data
public class Customer {
    private String id;
    private String name;
    private String email;
}

Billing Service Repository

@Repository
public interface InvoiceRepository extends JpaRepository<Invoice, String> {
    Page<Invoice> findByCustomerID(String customerId, Pageable pageable);
}

Billing Service OpenFeign Client

Adding the dependency:

<!-- https://mvnrepository.com/artifact/org.springframework.cloud/spring-cloud-starter-openfeign -->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
    <version>3.1.4</version>
</dependency>

Adding a rest client to our service in the openFeign package:

@FeignClient(name = "CUSTOMER-SERVICE")
// declare a rest client connected to CUSTOMER-SERVICE
public interface CustomerServiceRestClient {
    @GetMapping(path = "/api-v1/customers/{id}")
    Customer getCustomerById(@PathVariable String id);
    @GetMapping(path = "/api-v1/customers/")
    List<Customer> getCustomers();
}
  • Now our service will have the possibility to access the data of customer service via this interface functions.

  • This interface gives us the ability to manage the Customer-service data just like we do with JpaRepository, but via Rest (representational state transfer ) protocol.

  • So OpenFeign gives us that possibility in a easy way

Here when we call for example the function getCustomerById(), OpenFeign will just send an HTTP request to the CUSTOMER-SERVICE service (can retrieve it by the Discovery), and call the request on the path /api-v1/customers/{id}, that will return the serialized Customer to it, then it will return it a function returned Customer object....

To be able to use OpenFeign in our App we should enable it on the Application Main class. Just by adding @EnableFeignClients at the top of the Class

@EnableFeignClients
@SpringBootApplication
public class MyBillingServiceApplication {

    public static void main(String[] args) {
        SpringApplication.run(MyBillingServiceApplication.class, args);
    }

}

Billing Service DTOs

@Data
public class InvoiceRequestDTO {
    //private String id; // generated automatically , no need
    // private Date date; // generated automatically , no need
    private BigDecimal amount;
    private String customerId;

}
@Data
public class InvoiceResponseDTO {
    private String id;
    private Date date;
    private BigDecimal amount;
    private String customerId;
    private Customer customer;
}

Billing Service Mapping

Using mapstruct:

@Mapper(componentModel = "spring")
public interface InvoiceMapper {
    InvoiceResponseDTO toInvoiceResponse(Invoice invoice);
    Invoice toInvoice(InvoiceRequestDTO invoiceRequestDTO);
}

Billing Service Service layer

The Service

public interface InvoiceService {

    // list all
    List<InvoiceResponseDTO> getInvoicesList(int page, int size);
    // get one
    InvoiceResponseDTO getInvoiceById(String id);
    // insert
    InvoiceResponseDTO saveInvoice(InvoiceRequestDTO invoiceRequestDTO);
    // update
    InvoiceResponseDTO updateInvoice(String id, InvoiceRequestDTO invoiceRequestDTO);
    // delete
    boolean deleteInvoice(String id);
    // get customer's invoices
    List<InvoiceResponseDTO> getInvoicesListByCustomer(String customerId, int page, int size);

}

The Implementation

[Show code details]

@AllArgsConstructor
@Service
@Transactional
public class InvoiceServiceImp implements InvoiceService{

    private InvoiceRepository invoiceRepository;
    private InvoiceMapper invoiceMapper;

    private CustomerServiceRestClient customerServiceRestClient;

    @Override
    public List<InvoiceResponseDTO> getInvoicesList(int page, int size) {
        List<Invoice> invoiceList = invoiceRepository.findAll(PageRequest.of(page, size)).getContent();
        List<InvoiceResponseDTO> invoiceResponseDTOList = invoiceList.stream().map(
                invoice -> {
                    Customer customer = customerServiceRestClient.getCustomerById(invoice.getCustomerID());
                    if (customer == null) throw new RuntimeException(String.format("Can Not Find Customer with ID: %s", invoice.getCustomerID()));
                    invoice.setCustomer(customer);
                    return invoiceMapper.toInvoiceResponse(invoice);
                }
        ).collect(Collectors.toList());
        return invoiceResponseDTOList;
    }

    @Override
    public InvoiceResponseDTO getInvoiceById(String id) {
        Invoice invoice = invoiceRepository.findById(id).orElseThrow(() -> new RuntimeException(String.format("Invoice with ID: %s Not Found !", id)));
        Customer customer = customerServiceRestClient.getCustomerById(invoice.getCustomerID());
        if (customer == null) throw new RuntimeException(String.format("Can Not Find Customer with ID: %s", invoice.getCustomerID()));
        invoice.setCustomer(customer);
        return invoiceMapper.toInvoiceResponse(invoice);
    }

    @Override
    public InvoiceResponseDTO saveInvoice(InvoiceRequestDTO invoiceRequestDTO) {
        Invoice invoice = invoiceMapper.toInvoice(invoiceRequestDTO);
        invoice.setId(UUID.randomUUID().toString());
        invoice.setDate(new Date());
        // referential integrity check (validation)
        Customer customer = customerServiceRestClient.getCustomerById(invoiceRequestDTO.getCustomerId());
        if (customer == null) throw new RuntimeException(String.format("Can Not Find Customer with ID: %s", invoiceRequestDTO.getCustomerId()));
        invoice.setCustomer(customer);
        return invoiceMapper.toInvoiceResponse(invoiceRepository.save(invoice));
    }

    @Override
    public InvoiceResponseDTO updateInvoice(String id, InvoiceRequestDTO invoiceRequestDTO) {
        invoiceRepository.findById(id).orElseThrow(() -> new RuntimeException(String.format("Invoice with ID: %s Not Found !", id)));
        Invoice invoiceToSave = invoiceMapper.toInvoice(invoiceRequestDTO);
        invoiceToSave.setId(id);
        Customer customer = customerServiceRestClient.getCustomerById(invoiceRequestDTO.getCustomerId());
        if (customer == null) throw new RuntimeException(String.format("Can Not Find Customer with ID: %s", invoiceRequestDTO.getCustomerId()));
        invoiceToSave.setCustomer(customer);
        return invoiceMapper.toInvoiceResponse(invoiceRepository.save(invoiceToSave));
    }

    @Override
    public boolean deleteInvoice(String id) {
        invoiceRepository.findById(id).orElseThrow(() -> new RuntimeException(String.format("Invoice with ID: %s Not Found !", id)));
        invoiceRepository.deleteById(id);
        return true;
    }

    @Override
    public List<InvoiceResponseDTO> getInvoicesListByCustomer(String customerId, int page, int size) {
        List<Invoice> invoiceList = invoiceRepository.findByCustomerID(customerId, PageRequest.of(page, size)).getContent();
        List<InvoiceResponseDTO> invoiceResponseDTOList = invoiceList.stream().map(
                invoice -> {
                    Customer customer = customerServiceRestClient.getCustomerById(invoice.getCustomerID());
                    if (customer == null) throw new RuntimeException(String.format("Can Not Find Customer with ID: %s", invoice.getCustomerID()));
                    invoice.setCustomer(customer);
                    return invoiceMapper.toInvoiceResponse(invoice);
                }
        ).collect(Collectors.toList());
        return invoiceResponseDTOList;
    }
}

⚠️

  • In the case of Microservices , we should be careful withe the referential integrity
  • We should check (do validations), because the DB not one, and the 2 Dbs does not know about each other, so we should contrôl the relations ourselfs.
  • For example in this function, we check if the customer exists befoure add it to the Invoice
@Override
    public InvoiceResponseDTO saveInvoice(InvoiceRequestDTO invoiceRequestDTO) {
        Invoice invoice = invoiceMapper.toInvoice(invoiceRequestDTO);
        invoice.setId(UUID.randomUUID().toString());
        invoice.setDate(new Date());
        // referential integrity check (validation)
        Customer customer = customerServiceRestClient.getCustomerById(invoiceRequestDTO.getCustomerId());
        if (customer == null) throw new RuntimeException(String.format("Can Not Find Customer with ID: %s", invoiceRequestDTO.getCustomerId()));
        invoice.setCustomer(customer);
        return invoiceMapper.toInvoiceResponse(invoiceRepository.save(invoice));
    }

Billing Service Rest Controller

Code

@RestController
@RequestMapping(path = "/api-v1/invoices")
@AllArgsConstructor
public class InvoiceController {

    InvoiceService invoiceService;

    // list pagination
    @GetMapping("/")
    public List<InvoiceResponseDTO> getInvoicesList(@RequestParam(defaultValue = "0") int page,
                                                    @RequestParam(defaultValue = "10") int size){
        return invoiceService.getInvoicesList(page, size);
    }
    // list customer invoices
    @GetMapping("/customer/{customerId}")
    public List<InvoiceResponseDTO> getCustomerInvoices(@PathVariable String customerId,
                                                        @RequestParam(defaultValue = "0") int page,
                                                        @RequestParam(defaultValue = "10") int size){
        return invoiceService.getInvoicesListByCustomer(customerId, page, size);
    }

    // get one by id
    @GetMapping("/{invoiceId}")
    public InvoiceResponseDTO getInvoice(@PathVariable String invoiceId){
        return invoiceService.getInvoiceById(invoiceId);
    }
    // insert

    @PostMapping("/")
    public InvoiceResponseDTO insertInvoice(@RequestBody @Valid InvoiceRequestDTO invoiceRequestDTO){
        return invoiceService.saveInvoice(invoiceRequestDTO);
    }

    // update
    @PostMapping("/{invoiceId}")
    public InvoiceResponseDTO updateInvoice(@RequestBody @Valid InvoiceRequestDTO invoiceRequestDTO,
                                            @PathVariable String invoiceId){
        return invoiceService.updateInvoice(invoiceId, invoiceRequestDTO);
    }
    // delete
    @PostMapping("/{invoiceId}")
    public boolean deleteInvoice(@PathVariable String invoiceId){
        return invoiceService.deleteInvoice(invoiceId);
    }

}

Billing Service documentation

  • Adding the Dependency
<!-- https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-ui -->
<dependency>
    <groupId>org.springdoc</groupId>
    <artifactId>springdoc-openapi-ui</artifactId>
    <version>1.6.12</version>
</dependency>
  • Visiting http://localhost:8083/v3/api-docs to get the open api structure.
  • Visiting http://localhost:8083/swagger-ui/index.html to get the swagger Ui utility (testing interface).

Billing Service tests

Now to test our service we should complete the architecture, because the billing-service depends on the customer-service, and to now it, we should configure the Gateway and Descovery services of our application.

  • Before that let's run our App :
  • Exception : 🔥 🔥
...
Caused by: java.lang.IllegalStateException: Ambiguous mapping. Cannot map 'invoiceController' method 
me.elaamiri.mybillingservice.controllers.InvoiceController#updateInvoice(InvoiceRequestDTO, String)
to {POST [/api-v1/invoices/{invoiceId}]}: There is already 'invoiceController' bean method
....
  • Problem: I'm mapping POST method to more then a function in the controller :
// update
    @PostMapping("/{invoiceId}")
    public InvoiceResponseDTO updateInvoice(@PathVariable String invoiceId,@RequestBody @Valid InvoiceRequestDTO invoiceRequestDTO){
        return invoiceService.updateInvoice(invoiceId, invoiceRequestDTO);
    }
    // delete
    @PostMapping("/{invoiceId}")
    public boolean deleteInvoice(@PathVariable String invoiceId){
        return invoiceService.deleteInvoice(invoiceId);
    }

Discovery service

  • Is is a technical service (un service technique).
  • Needs just the dependency :
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
  • We need to enable Eureka server in the Application main point
@EnableEurekaServer
@SpringBootApplication
public class DiscoveryServiceApplication {

	public static void main(String[] args) {
		SpringApplication.run(DiscoveryServiceApplication.class, args);
	}

}
  • We need to add some properties also
server.port=8585
# do not register itself as a client
eureka.client.fetch-registry=false
# Does not register itself in the service registry
eureka.client.register-with-eureka=false
  • Visiting http://localhost:8585/

7

  • Now we can enable discovery client on both of our services
spring.cloud.discovery.enabled=true
  • When we execute our 2 services (Customers + billing), we supposed to find them on the Eureka interface but => 🔥 THEY DO NOT 🔥

  • Problem : Using the port 8585 for eureka server which not the default one, used by the other services to connect the eureka server.

  • Solution: Use the port 8761 as server port in the properties of the eureka service.

  • In this case we can visualise our micro-services on the eureka interface

  • There is a properity for the services to prefer the IP address to register to the discovery instead of the name :

eureka.instance.prefer-ip-address=true
  • In this cas the service when it is regestred to eureka, will register service name + IP @ + Port

  • Now Our services are connected successfully (Just be careful about services names).

  • Also be careful about the fields names, they must be the same between the DTOs and the Entities, so we avoid the ambiguity in the mapping phase?

  • Testing a POST on : 'http://localhost:8083/api-v1/invoices/'

Request Body

{
  "amount":7778,
  "customerID": "96d2de30-a0dd-4243-8806-b6748e49e763"
}

Response

{
  "id": "476f87d1-f0ad-4431-bf2d-e623d4dae5b9",
  "date": "2022-10-29T00:12:42.303+00:00",
  "amount": 7778,
  "customer": {
    "id": "96d2de30-a0dd-4243-8806-b6748e49e763",
    "name": "Ahmed",
    "email": "Ahmed_email@gmail.com"
  }
}
  • In this case eureka helped us in connection needed by OpenFiegn, because we used the name of the service on the CustomerRestClient, but we can even do that without OpenFiegn, in that case we must use the service URL (explicitly).
  • Using the name is more intersted because in case we use more then 1 instance of CUSTOMER-SERVICE, the load balancer will control which instance we use in a dynamic way..

Gateway service

  • In the precedent sections we connected to our services via link of each one, now we will get things more dynamic => Routage (Routing).
  • The gateway, is a technical micro-service.
  • We just need to add the dependencies:
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-gateway</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
  • Configuring the dynamic routing, by adding the class :
@Configuration
public class GatewayConfig {
    @Bean
    DiscoveryClientRouteDefinitionLocator discoveryClientRouteDefinitionLocator(ReactiveDiscoveryClient reactiveDiscoveryClient,
    DiscoveryLocatorProperties discoveryLocatorProperties){
        return new DiscoveryClientRouteDefinitionLocator(reactiveDiscoveryClient, discoveryLocatorProperties);
    }
}
  • By this config, we are telling to Soring cloud Gateway: everytime a request arrives,

    1. find the name of the service from the path
    2. Contact the discovery service to give you the IP @ + Port of the service (or instance of it ..)
    3. Dispatch towards the service, and get data
  • We need also some properties:

server.port=8888
spring.application.name=GATEWAY-SERVICE
## the gateway also should be registred in the discovery server
eureka.client.register-with-eureka=true

Here is all our services registred on Eureka

8

  • Now I can access my microservice via the Gateway :
  • Visiting : http://localhost:8888/GATEWAY-SERVICE/CUSTOMER-SERVICE/api-v1/customers/
  • Result:
[{"id":"96d2de30-a0dd-4243-8806-b6748e49e763","name":"Ahmed","email":"Ahmed_email@gmail.com"},{"id":"6ad2bed8-6e10-4845-a6a1-fe57ba891444","name":"Essadeq","email":"Essadeq_email@gmail.com"}]
  • How that works :

9

Refactoring

Handling Exceptions

  • Creating Exceptions (for example for the customer service): LINK TO FILES
package me.elaamiri.customerservice.Exceptions;

public class CustomerNotFountException extends RuntimeException{
    public CustomerNotFountException(String message) {
        super(message);
    }
}
  • To handle this exceptions from the controller we can add a function:

Using my custom Response Class

@ExceptionHandler(Exception.class)
    public ExceptionResponse exceptionHandler(Exception exception){
        ExceptionResponse exceptionResponse = new ExceptionResponse();
        exceptionResponse.setHttpStatus(HttpStatus.NOT_FOUND);
        exceptionResponse.setMessage(exception.getMessage());
        return exceptionResponse;
    }

Or using ResponseEntity

 @ExceptionHandler(Exception.class)
    public ResponseEntity<String> exceptionHandler(Exception exception){

        return new ResponseEntity<>(exception.getMessage(), HttpStatus.INTERNAL_SERVER_ERROR);
    }

Dockerizing our micro-services

Dockerise Using Docker File and Docker compose

  1. First create a docker file in our root application folder and name the file as Dockerfile . It is important to follow this naming convention as it is how Spring will locate this file.
  • The Docker File looks like-
FROM openjdk:8-jdk-alpine
ARG JAR_FILE=target/*.jar
COPY ${JAR_FILE} app.jar
ENTRYPOINT ["java","-jar","/app.jar"]
  • FROM: Specifies the image that has to be downloaded (base image)
  • ARG: Specifies an argument
  • COPY: Copies the JAR_FILE into the container as app.jar
  • ENTRYPOINT: Specifies the command which will be executed first
  1. Next, run the below command to generate the jar file of your application.
  • For Maven-
$ mvn clean install
  • For Gradle-
$ gradle build
  • We can execute them by the utility provided in the IDE (IDE will use its integreted version of Gradle or Maven).

  • Faced the Exception :

- class lombok.javac.apt.LombokProcessor (in unnamed module @0x3946075) cannot access class com.
- sun.tools.javac.processing.JavacProcessingEnvironment (in module jdk.compiler) because module jdk.
- compiler does not export com.sun.tools.javac.processing to unnamed module @0x3946075
  • Solution update lombok version 1.18.16 -> 1.18.22

  • Faced the Exception

- There are test failures.
Please refer to 
C:\Users\elaam\IdeaProjects\microservices-practical-activity-docker\my-billing-service\target\sure
fire-reports for the individual test results.
Please refer to dump files (if any exist) [date].dump, [date]-jvmRun[N].dump and [date].dumpstream.
  • Maven explains it
-------------------------------------------------------------------------------
Test set: me.elaamiri.mybillingservice.MyBillingServiceApplicationTests
-------------------------------------------------------------------------------
Tests run: 1, Failures: 0, Errors: 1, Skipped: 0, Time elapsed: 13.512 s <<< FAILURE! - in me.elaamiri.mybillingservice.MyBillingServiceApplicationTests
contextLoads  Time elapsed: 0.001 s  <<< ERROR!
java.lang.IllegalStateException: Failed to load ApplicationContext
Caused by: java.lang.IllegalStateException: Failed to execute CommandLineRunner
Caused by: feign.FeignException$ServiceUnavailable: [503] during [GET] to [http://CUSTOMER-SERVICE/api-v1/customers/96d2de30-a0dd-4243-8806-b6748e49e763] [CustomerServiceRestClient#getCustomerById(String)]: [Load balancer does not contain an instance for the service CUSTOMER-SERVICE]
  • Just removed the CommandRunner code there as solution/
  • After that all jar files of our services are generated,
  1. Now create the docker image by executing the below command.
  • For Maven-
$ docker build -t <image-name>:<image-tag> .
  • For Gradle-
$ docker build --build-arg JAR_FILE=build/libs/*.jar -t tag-name/image-name .
  • Here are our images > docker images:

10

To wrap up the above steps, the build will create a spring user and a spring group to run the application. It will later COPY the project JAR file into the container as "app.jar" that will be executed in the ENTRYPOINT.

  1. Let's create the docker compose to run everything :
  • Tagging images to push them

11

12

❌ => File corrected version below

version: '3'
services:
  discovery-service:
    image: elaamiri/discovery-service:first-version
    ports:
      - 8761:8761
  gateway-service:
    image: elaamiri/gateway-service:first-version
    ports:
      - 8888:8888
    depends_on:
      - discovery-service
  customer-service:
    image: elaamiri/customer-service:first-version
    ports:
      - 8082:8082
    depends_on:
      - discovery-service
  my-billing-service:
    image: elaamiri/my-billing-service:first-version
    ports:
      - 8083:8083
    depends_on:
      - discovery-service
  • Lunchinf the docker-compose
> docker-compose -f docker-compose.yml up
  • Here are our containers

13

  • Result

  • I can access my services in a separated way correctly

  • Can't find my services on the Discovery service (UI interface)

  • Can't access my services via gateway service

  • Caused by: java.net.ConnectException: Connection refused Exceptionon the servicess

  • Added eureka.client.serviceUrl.defaultZone=http://discovery-service:8761/eureka/ as property but the error still there.

  • Error : it seems like I miswrite the property,

  • To avoid build and rebuild the image everytime, we should use environement variables in the docker compose file.

  • Here is the new version of our docker-compose.yml

version: '3'
services:
  discovery-service:
    image: elaamiri/discovery-service:first-version
    ports:
      - 8761:8761
    
  gateway-service:
    image: elaamiri/gateway-service:first-version
    ports:
      - 8888:8888
    depends_on:
      - discovery-service
    environment:
      - eureka.client.service-url.defaultZone=http://discovery-service:8761/eureka
  customer-service:
    image: elaamiri/customer-service:first-version
    ports:
      - 8082:8082
    environment:
      - eureka.client.service-url.defaultZone=http://discovery-service:8761/eureka
    depends_on:
      - discovery-service
      - gateway-service
  my-billing-service:
    image:  elaamiri/my-billing-service:1.0.1
    ports:
      - 8083:8083
    environment:
      - eureka.client.service-url.defaultZone=http://discovery-service:8761/eureka
    depends_on:
      - discovery-service
      - gateway-service
      - customer-service
  • Our services are registred to discovery service.

14

  • Now Visiting http://localhost:8888/GATEWAY-SERVICE/CUSTOMER-SERVICE/api-v1/customers/
  • Gives us as result:
[{"id":"6bd53d1a-45fc-4bb5-9ceb-08254e8f4ca3","name":"Ahmed","email":"Ahmed_email@gmail.com"},{"id":"73953e0e-16c9-4174-b0cb-e2f369a9af14","name":"Essadeq","email":"Essadeq_email@gmail.com"}]
  • To stop the services :
> docker-compose -f docker-compose.yml down

Dockerise Using Jib

REF: https://medium.com/the-sixt-india-blog/dockerise-a-spring-boot-application-bdfce1d6eb15

In other ways ...

REF : https://spring.io/guides/topicals/spring-boot-docker/


  • This is a Synchronous communication based architecture,

What next ?

  • Using Brockers to make an Asynchronous communication based architecture.
  • Adding Config-service
  • Adding Security layer
  • Dockerizing using Kubernetes