From 9f850bb90d5fdcbf66414d9234a0c6416e8c3799 Mon Sep 17 00:00:00 2001 From: Ibrahim Abou Elenein Date: Sun, 26 May 2024 02:28:12 +0300 Subject: [PATCH] Update README.md (#115) --- README.md | 191 +++++++++++++++++++++++++++++++++++++++++++----------- 1 file changed, 154 insertions(+), 37 deletions(-) diff --git a/README.md b/README.md index bdc9eaa..016788b 100644 --- a/README.md +++ b/README.md @@ -58,32 +58,58 @@ WorkUp is a microservices-based application that allows freelancers and clients ## ๐Ÿงฉ Components -Swarm on the machines -![image](https://github.com/Ahmad45123/workup/assets/35760882/27bcc9e8-316c-4248-b470-b975a6411962) -![image](https://github.com/Ahmad45123/workup/assets/35760882/4d2222b6-127d-4fb3-9f04-150b3f495daa) +
+ + Swarm on the machines + + + ![image](https://github.com/Ahmad45123/workup/assets/35760882/27bcc9e8-316c-4248-b470-b975a6411962) + ![image](https://github.com/Ahmad45123/workup/assets/35760882/4d2222b6-127d-4fb3-9f04-150b3f495daa) + +
-### ๐Ÿ—๏ธ Microservices - -All microservices are implemented using Java Spring Boot โ˜•. They consume messages from the RabbitMQ message broker ๐Ÿฐ and respond through RabbitMQ as well. Requests are cached in Redis, so if the same request is sent more than once, there is no need to recompute the response every time as it can be retrieved directly from the cache ๐Ÿ—ƒ๏ธ. All the microservices are stateless, as authentication and authorization are handled by the web server ๐Ÿ”. In some cases, a microservice would have to communicate with another one to complete a certain functionality, which is done through RabbitMQ. Every microservice can be scaled up or down independently of other microservices to adapt to the amount of incoming traffic ๐Ÿ“ˆ. Every microservice has its database that is shared by all the instances of the same microservice but cannot be accessed by other microservices. All the microservices have multithreading, where a thread pool is created, and every request is assigned to a thread from that pool to allow asynchronous computation. -#### ๐Ÿ‘ฅ Users Microservice +### ๐Ÿ—๏ธ Microservices -This microservice handles user-related operations, like updating, fetching, deleting, and creating profiles, register & login ๐Ÿ‘ค. In the case of register or login, a JWT is created and returned in the response, which will be used by the client to authorize future communications with the system ๐Ÿ”‘. The database used for this microservice is MongoDB ๐Ÿƒ, as it achieves horizontal scalability, is highly available, and provides higher performance for reads and writes than relational databases. MongoDB 4.0 and later supports ACID transactions to some extent, which was enough for the users' microservice use case. +- All microservices are implemented using Java Spring Boot โ˜•. +- They consume messages from the RabbitMQ message broker ๐Ÿฐ and respond through RabbitMQ as well. +- Requests are cached in Redis, so if the same request is sent more than once, there is no need to recompute the response every time as it can be retrieved directly from the cache ๐Ÿ—ƒ๏ธ. +- All the microservices are stateless, as authentication and authorization are handled by the web server. +- In some cases, a microservice would have to communicate with another one to complete a certain functionality, which is done through RabbitMQ. Every microservice can be scaled up or down independently of other microservices to adapt to the amount of incoming traffic. +- Every microservice has its database that is shared by all the instances of the same microservice but cannot be accessed by other microservices. -#### ๐Ÿ’ณ Payments Microservice +
+ + ๐Ÿ‘ฅ Users Microservice + + This microservice handles user-related operations, like updating, fetching, deleting, and creating profiles, register & login ๐Ÿ‘ค. In the case of register or login, a JWT is created and returned in the response, which will be used by the client to authorize future communications with the system ๐Ÿ”‘. The database used for this microservice is MongoDB ๐Ÿƒ, as it achieves horizontal scalability, is highly available, and provides higher performance for reads and writes than relational databases. MongoDB 4.0 and later supports ACID transactions to some extent, which was enough for the users' microservice use case. +
+
+ + ๐Ÿ’ณ Payments Microservice + + This microservice handles payment-related requests ๐Ÿ’ณ. Both freelancers and clients have wallets, which they can add or withdraw money from. Both of them have a history of transactions, and freelancers can issue payment requests that are then paid by the clients, and the money is deposited into the freelancer's wallet. When the payment is completed, the contracts service is notified that the contract associated with this payment should be marked as done. The DB used for this service is PostgreSQL ๐Ÿ˜, as payments require lots of ACID transactions in addition to strict consistency, which is achieved by relational databases. +
-This microservice handles payment-related requests ๐Ÿ’ณ. Both freelancers and clients have wallets, which they can add or withdraw money from. Both of them have a history of transactions, and freelancers can issue payment requests that are then paid by the clients, and the money is deposited into the freelancer's wallet. When the payment is completed, the contracts service is notified that the contract associated with this payment should be marked as done. The DB used for this service is PostgreSQL ๐Ÿ˜, as payments require lots of ACID transactions in addition to strict consistency, which is achieved by relational databases. -#### ๐Ÿ”ง Jobs Microservice +
+ + ๐Ÿ”ง Jobs Microservice + + This microservice handles jobs and proposal requests ๐Ÿ“. Clients can create jobs, and view, and accept proposals for their jobs. Freelancers can browse jobs and search for them using keywords ๐Ÿ”. They can submit a proposal to any job, specifying the milestones that will achieve that job and any extra attachments. When a client accepts a proposal, the contracts service is notified about it to initiate a contract for that job ๐Ÿ“œ. The used DB for this microservice is Cassandra, as it can be scaled horizontally easily and is highly available and fault-tolerant, thanks to its peer-to-peer decentralized architecture. There is no need to have a relational DB for jobs, since there are not many joins performed, and there is no need for strong consistency and strict schema. However, high availability is critical, as most of the time users will be browsing jobs, which implies high traffic on the DB. +
-This microservice handles jobs and proposal requests ๐Ÿ“. Clients can create jobs, and view, and accept proposals for their jobs. Freelancers can browse jobs and search for them using keywords ๐Ÿ”. They can submit a proposal to any job, specifying the milestones that will achieve that job and any extra attachments. When a client accepts a proposal, the contracts service is notified about it to initiate a contract for that job ๐Ÿ“œ. The used DB for this microservice is Cassandra, as it can be scaled horizontally easily and is highly available and fault-tolerant, thanks to its peer-to-peer decentralized architecture. There is no need to have a relational DB for jobs, since there are not many joins performed, and there is no need for strong consistency and strict schema. However, high availability is critical, as most of the time users will be browsing jobs, which implies high traffic on the DB. +
+ + ๐Ÿ“œ Contracts Microservice + + This microservice is responsible for contract-related logic ๐Ÿ“‘. It handles the termination and creation of contracts. Freelancers can update their progress in a milestone of their contracts, while clients can add evaluation to every milestone. Cassandra was used as a database for this microservice for the same reasons as the jobs microservice: scalability, high availability, and fault tolerance. +
-#### ๐Ÿ“œ Contracts Microservice -This microservice is responsible for contract-related logic ๐Ÿ“‘. It handles the termination and creation of contracts. Freelancers can update their progress in a milestone of their contracts, while clients can add evaluation to every milestone. Cassandra was used as a database for this microservice for the same reasons as the jobs microservice: scalability, high availability, and fault tolerance. ### ๐Ÿ“ฌ Message Queues @@ -104,14 +130,19 @@ All endpoints require authentication using a bearer token. Include the token in Authorization: Bearer ``` -#### Get Job +
+ + Get Job + Description: Retrieves details of a specific job. - URL: /api/v1/jobs/{job_id} - Method: GET + Request Parameters - job_id (path parameter): The ID of the job to retrieve. - Example Request + ```bash GET /api/v1/jobs/7ddda13b-8221-4766-983d-9068a6592eba Authorization: Bearer @@ -130,7 +161,14 @@ Response ``` - 404 Not Found: If the job with the specified ID does not exist. -#### Get Proposals +
+ +
+ + Get Proposals + + + Description: Retrieves all proposals for a specific job. - URL: /api/v1/jobs/{job_id}/proposals - Method: GET @@ -157,8 +195,12 @@ Response ``` - 404 Not Found: If the job with the specified ID does not exist - -#### Get Contract +
+
+ + Get Contract + + Description: Retrieves details of a specific contract. - URL: /api/v1/contracts/{contract_id} @@ -187,9 +229,14 @@ Response - 404 Not Found: If the contract with the specified ID does not exist. +
- -#### Create Proposal +
+ + Create Proposal + + + Description: Creates a new proposal for a specific job. - URL: /api/v1/jobs/{job_id}/proposals @@ -227,9 +274,13 @@ Response ``` - 404 Not Found: If the job with the specified ID does not exist. +
- -#### Get Proposal +
+ + Get Proposal + + Description: Retrieves details of a specific proposal. - URL: /api/v1/proposals/{proposal_id} @@ -254,12 +305,14 @@ Response } ``` - - - - 404 Not Found: If the proposal with the specified ID does not exist. +
-##### Update Proposal +
+ + Update Proposal + + Description: Updates an existing proposal. - URL: /api/v1/proposals/{proposal_id} @@ -296,6 +349,7 @@ Response } ``` - 404 Not Found: If the proposal with the specified ID does not exist. +
### ๐ŸŽ›๏ธ Controller @@ -326,6 +380,12 @@ A media server that serves static resources. For all static resources, this server will attempt to return a relevant resource ๐Ÿ“, or else if the resource does not exist, it will return a default 'placeholder' resource ๐Ÿ–ผ๏ธ. This prevents clients from having no resource to display at all; clients can make use of this media server's 'describe' endpoint to learn about what resources are available ๐Ÿ“‹. +
+ + Get resource + + + #### `GET` /static/icons/:icon.png Returns an icon based on the filename. @@ -352,7 +412,13 @@ curl --request GET \ --url http://path_to_server/static/resume/resume.pdf ``` +
+
+ + Available groups + + #### Describe #### `GET` /describe @@ -395,10 +461,15 @@ curl --request GET \ "files": [] } ``` +
-#### Upload +
+ + Upload + + -Upload and convert media to any of the given static resource group. +Upload and convert media to any of the given static resource groups. All upload routes are protected by basic HTTP auth. The credentials are defined by ENV variables `UPLOAD_USER` and `UPLOAD_PASSWORD`. @@ -418,17 +489,19 @@ curl --location 'http://path-to-server/upload/resume/' \ ```json { "success": true, - "path": "/static/resume/aboueleyes-reume-2.pdf" + "path": "/static/resume/aboueleyes-resume-2.pdf" } ``` A resource at `http://path_to_server/static/resume/aboueleyes-reume-2.pdf` will now be available. +
+ ## ๐Ÿš€ Deployment ![Digital Ocean](https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcQfQxeW7i7mdvQqs7IdiokF0IIaenP9OvqO7NZLVNca&s) -We were able to setup a tenant on [Digital ocean](https://try.digitalocean.com/cloud/?utm_campaign=emea_brand_kw_en_cpc&utm_adgroup=Misspellings&_keyword=digitalocean%27&_device=c&_adposition=&utm_content=conversion&utm_medium=cpc&utm_source=google&gad_source=1&gclid=CjwKCAjwr7ayBhAPEiwA6EIGxFUoqeEw6G9c-Vh1FvlsGCa6kaL7uDHwQMuVojoNVrMNXKcXnJ906BoCFVwQAvD_BwE) ๐ŸŒ Where we can create new intance(Droplet) ๐Ÿ’ง from the Controller as explained above and feeding scripts in the startup of this instance to join docker swarm ๐Ÿณ of other machines. +We were able to set up a tenant on [Digital ocean](https://try.digitalocean.com/cloud/?utm_campaign=emea_brand_kw_en_cpc&utm_adgroup=Misspellings&_keyword=digitalocean%27&_device=c&_adposition=&utm_content=conversion&utm_medium=cpc&utm_source=google&gad_source=1&gclid=CjwKCAjwr7ayBhAPEiwA6EIGxFUoqeEw6G9c-Vh1FvlsGCa6kaL7uDHwQMuVojoNVrMNXKcXnJ906BoCFVwQAvD_BwE) ๐ŸŒ Where we can create new instance (Droplet) ๐Ÿ’ง from the Controller as explained above and feeding scripts in the startup of this instance to join docker swarm ๐Ÿณ of other machines. We can monitor the performance of each running container on each instance using [Portainer](https://www.portainer.io/) ๐Ÿ“Š where we can manually run more containers of the same service, and configure it to auto-scale when the load is above the threshold. @@ -462,13 +535,20 @@ deploy: cpus: '0.50' ``` -## Configuration +
+ + Configuration + + + | Setting | Value | Description | | --- | --- | --- | | `swarm.autoscaler` | `true` | Required. This enables autoscaling for a service. Anything other than `true` will not enable it | | `swarm.autoscaler.minimum` | Integer | Optional. This is the minimum number of replicas wanted for a service. The autoscaler will not downscale below this number | | `swarm.autoscaler.maximum` | Integer | Optional. This is the maximum number of replicas wanted for a service. The autoscaler will not scale up past this number | +
+ ## Load Balancing We used a round-robin-based load balancing ๐Ÿ”„ that is handled by Docker in the Swarm ๐Ÿณ. Simply, it sends the first request to the first instance of the app (not for a machine) and the next request to the next instance, and so on until all instances have received a request. Then, it starts again from the beginning. ๐Ÿš€ @@ -479,36 +559,73 @@ We used a round-robin-based load balancing ๐Ÿ”„ that is handled by Docker in the To test our app functionality we created functional testing for each service on its own to test functionality in isolation as well as testing its functionality in integration with other services for example [here](services\payments\src\test\java\com\workup\payments\PaymentsApplicationTests.java) is the tests implemented for Payments service.๐Ÿงช๐Ÿ”ง + ### โš–๏ธ Load Balancing +
+ + Jmeter + + ![jmeter](https://i0.wp.com/cdn-images-1.medium.com/max/800/1*KeuQ7uNalz2l4rBOyPAUpg.png?w=1180&ssl=1) We used JMeter to load test our app we configured it to simulate thousands of users' requests and the number grew gradually over the span of 10 seconds. here are some results for different endpoints. Here are a few examples of endpoint performance. +
-#### Login Performance +
+ + Login Performance + + ![login command](https://media.discordapp.net/attachments/1210626240986226741/1241779393614057562/image.png?ex=664ebc6e&is=664d6aee&hm=f33b046fa5bb5117cf1b75dd70d34f3b6bbcb2e11f6e639e3dc1bf3802ff7b79&=&format=webp&quality=lossless) -#### Create Job +
-![Create Job](https://media.discordapp.net/attachments/1210626240986226741/1241782178824716339/image.png?ex=664ebf06&is=664d6d86&hm=673ee8ac6d1d218aa49aa0491e96f11dd3f536a8f69d6fb69d8e47bf14863a84&=&format=webp&quality=lossless) +
+ + Create Job + + + ![Create Job](https://media.discordapp.net/attachments/1210626240986226741/1241782178824716339/image.png?ex=664ebf06&is=664d6d86&hm=673ee8ac6d1d218aa49aa0491e96f11dd3f536a8f69d6fb69d8e47bf14863a84&=&format=webp&quality=lossless) + +
-#### Get Jobs +
+ + Get Jobs + + ![get jobs](https://media.discordapp.net/attachments/1210626240986226741/1241787039603490858/image.png?ex=664ec38d&is=664d720d&hm=61b2c43df718b627191b90be4c24ff6d870e44b86294546b73da0f566015cdcc&=&format=webp&quality=lossless) -#### Get Contracts +
+
+ + Get Contracts + + ![get contracts](https://media.discordapp.net/attachments/1210626240986226741/1241803839947145387/image.png?ex=664ed333&is=664d81b3&hm=2ab4e1c58e3224a0c0f6ef8e4eee08330fb8e68e914ef2e3f586bc16f41d0b90&=&format=webp&quality=lossless) +
+
+ + Locust + + ![locust](https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcTOchmJvYjMucoFZDd2NVX6uLPNci5uEhBzF1vHDJMJ0Q&s) -We alos used python [locust](https://locust.io/) for load testing. check the test file [here](https://github.com/Ahmad45123/workup/blob/main/locustfile.py). Here is the the results of this load test +We also used `Python` [locust](https://locust.io/) for load testing. check the test file [here](https://github.com/Ahmad45123/workup/blob/main/locustfile.py). Here are the results of this load test ![requeststats](https://media.discordapp.net/attachments/1210626240986226741/1242789353558769664/image.png?ex=664f1d47&is=664dcbc7&hm=c6838d6f086a3f04688f7189fa042a0f3d07dee9026f389f804ef56fad88be84&=&format=webp&quality=lossless&width=863&height=676) ![charts](https://media.discordapp.net/attachments/1210626240986226741/1242789434923946076/image.png?ex=664f1d5b&is=664dcbdb&hm=d551fd87c5ebc96de32e9119ff41596bf54de326491ce3007f4a6f695865f1fb&=&format=webp&quality=lossless&width=861&height=676) +
+ + + ## License This project is licensed under the MIT License. See the [LICENSE](LICENSE) file for more details.