Compare commits
	
		
			2 Commits
		
	
	
		
			3dc8131c33
			...
			front_test
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a1322c64ca | |||
| 8394bf02f2 | 
| @@ -1,18 +0,0 @@ | |||||||
| name: Format |  | ||||||
|  |  | ||||||
| on: [ push, pull_request ] |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|  |  | ||||||
|   formatting: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 # v2 minimum required |  | ||||||
|       - uses: actions/setup-java@v4 |  | ||||||
|         with: |  | ||||||
|           distribution: 'temurin' # See 'Supported distributions' for available options |  | ||||||
|           java-version: '21' |  | ||||||
|  |  | ||||||
|       - uses: axel-op/googlejavaformat-action@v3 |  | ||||||
|         with: |  | ||||||
|           args: "--set-exit-if-changed --skip-sorting-imports --skip-reflowing-long-strings --aosp -n" |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| name: Build |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|  |  | ||||||
|     - name: Load .env file |  | ||||||
|       uses: xom9ikk/dotenv@v2.3.0 |  | ||||||
|       with: |  | ||||||
|         path: ./config/ |  | ||||||
|         mode: dev |  | ||||||
|         load-mode: strict |  | ||||||
|  |  | ||||||
|     - name: Setup Java |  | ||||||
|       uses: actions/setup-java@v4 |  | ||||||
|       with: |  | ||||||
|         distribution: 'temurin' |  | ||||||
|         java-version: 21 |  | ||||||
|      |  | ||||||
|     - name: Setup Gradle |  | ||||||
|       uses: gradle/actions/setup-gradle@v4 |  | ||||||
|       with: |  | ||||||
|         cache-disabled: true # Once the code has been pushed once in main, this should be reenabled.  |  | ||||||
|  |  | ||||||
|     - name: init gradle |  | ||||||
|       working-directory: ./MyINPulse-back/ |  | ||||||
|       run: ./gradlew build # todo: run test, currently fail because no database is present  |  | ||||||
| @@ -1,24 +0,0 @@ | |||||||
| name: CI |  | ||||||
| on: push |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout code |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|  |  | ||||||
|       - name: Install everything |  | ||||||
|         working-directory: ./front/MyINPulse-front |  | ||||||
|         run: npm i |  | ||||||
|        |  | ||||||
|       - name: Run ESLint |  | ||||||
|         working-directory: ./front/MyINPulse-front |  | ||||||
|         run: npx eslint |  | ||||||
|  |  | ||||||
|       - name: Run prettier |  | ||||||
|         working-directory: ./front/MyINPulse-front |  | ||||||
|         run: npx prettier src --check |  | ||||||
|  |  | ||||||
|       - name: Build frontend |  | ||||||
|         working-directory: ./front/MyINPulse-front |  | ||||||
|         run: npm run build |  | ||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,4 +2,3 @@ | |||||||
| .idea | .idea | ||||||
| keycloak/CAS/target | keycloak/CAS/target | ||||||
| docker-compose.yaml | docker-compose.yaml | ||||||
| postgres/data |  | ||||||
| @@ -1,741 +0,0 @@ | |||||||
| openapi: 3.0.3 |  | ||||||
| info: |  | ||||||
|   title: MyInpulse Backend Api |  | ||||||
|   description: this document servers as a documentation for the backend api.   |  | ||||||
|   version: 0.0.0 |  | ||||||
|  |  | ||||||
| tags: |  | ||||||
|   - name: Entrepreneurs API |  | ||||||
|     description: La partie de l'api dédiée aux entrepreneurs |  | ||||||
|   - name: Admin API |  | ||||||
|     description: La partie de l'api dédiée aux entrepreneurs |  | ||||||
|   - name: Shared API |  | ||||||
|     description: La partie de l'api dédiée aux entrepreneurs et admins |  | ||||||
|  |  | ||||||
|  |  | ||||||
| components: |  | ||||||
|   schemas: |  | ||||||
|     user: |  | ||||||
|       type: object |  | ||||||
|       properties: |  | ||||||
|         nom: |  | ||||||
|           type: string |  | ||||||
|         prenom: |  | ||||||
|           type: string |  | ||||||
|         email: |  | ||||||
|           type: string |  | ||||||
|           example: "example@exmaple.com" |  | ||||||
|         secondaryEmail: |  | ||||||
|           type: string |  | ||||||
|           example: "example@exmaple.com" |  | ||||||
|         tel: |  | ||||||
|           type: string |  | ||||||
|           example: "0612345678" |  | ||||||
|     user-entrepreneur: |  | ||||||
|       type: object |  | ||||||
|       properties: |  | ||||||
|         user: |  | ||||||
|           $ref: "#/components/schemas/user" |  | ||||||
|         entrepreneur: |  | ||||||
|           type: object |  | ||||||
|           properties: |  | ||||||
|             ecole:  |  | ||||||
|               type: string |  | ||||||
|               example: "enseirb" |  | ||||||
|             filiere: |  | ||||||
|               type: string |  | ||||||
|               example: "info" |  | ||||||
|             status: |  | ||||||
|               type: boolean |  | ||||||
|               example: false |  | ||||||
|     user-admin: |  | ||||||
|       type: object |  | ||||||
|       properties: |  | ||||||
|         admin: |  | ||||||
|           $ref: "#/components/schemas/user" |  | ||||||
|  |  | ||||||
|   securitySchemes: |  | ||||||
|       MyINPulse: |  | ||||||
|         type: oauth2 |  | ||||||
|         flows: |  | ||||||
|           implicit: |  | ||||||
|             authorizationUrl: https://petstore3.swagger.io/oauth/authorize |  | ||||||
|             scopes: |  | ||||||
|               MyINPulse-admin: Administrateur |  | ||||||
|               MyINPulse-entrepreneur: Utilisateur |  | ||||||
|  |  | ||||||
| paths: |  | ||||||
|  |  | ||||||
| #          _    ____  __  __ ___ _   _      _    ____ ___  |  | ||||||
| #         / \  |  _ \|  \/  |_ _| \ | |    / \  |  _ \_ _| |  | ||||||
| #        / _ \ | | | | |\/| || ||  \| |   / _ \ | |_) | |  |  | ||||||
| #       / ___ \| |_| | |  | || || |\  |  / ___ \|  __/| |  |  | ||||||
| #      /_/   \_\____/|_|  |_|___|_| \_| /_/   \_\_|  |___| |  | ||||||
| # |  | ||||||
|  |  | ||||||
|   /admin/projects: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne la liste of projets associés à l'admin |  | ||||||
|       tags:  |  | ||||||
|         - Admin API  |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|       description: |  | ||||||
|         JSON array of who's elements are objects containing necessary information for the view  |  | ||||||
|         (project name, entrepreneur names, etc..) |  | ||||||
|         of the projects an admin is watching over. |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/json: |  | ||||||
|               schema: |  | ||||||
|                 type: array |  | ||||||
|                 items: |  | ||||||
|                   type: object |  | ||||||
|                   properties: |  | ||||||
|                     name: |  | ||||||
|                       type: string |  | ||||||
|                     E_names: |  | ||||||
|                       type: string |  | ||||||
|                       description: entrepreneur names |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is missing or invalid |  | ||||||
|   /admin/projects/pending/decision: |  | ||||||
|     post: |  | ||||||
|       summary: valider un projet en attente de validation |  | ||||||
|       tags: |  | ||||||
|         - Admin API |  | ||||||
|       description: |  | ||||||
|         if the request is accepted the status of the |  | ||||||
|         project is changed to ongoing, entrepreneur |  | ||||||
|         account is confirmed and the project is linked |  | ||||||
|         to the admin accepting the request and the |  | ||||||
|         entrepreneur requesting it. Else the pending |  | ||||||
|         project and user info are deleted. |  | ||||||
|       security: |  | ||||||
|           - MyINPulse: |  | ||||||
|               - MyINPulse-admin |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 pedingProjectId:  |  | ||||||
|                   type: integer |  | ||||||
|                 decision: |  | ||||||
|                   type: boolean |  | ||||||
|                  |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|              |  | ||||||
|   /admin/projects/add: |  | ||||||
|     post: |  | ||||||
|       summary: Ajout manuel d'un projet |  | ||||||
|       description: |  | ||||||
|         Adds a project with the |  | ||||||
|         inputed details |  | ||||||
|       tags: |  | ||||||
|         - Admin API |  | ||||||
|       security: |  | ||||||
|           - MyINPulse: |  | ||||||
|               - MyINPulse-admin |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 name: |  | ||||||
|                   type: string |  | ||||||
|                 founder: |  | ||||||
|                   $ref: "#/components/schemas/user-entrepreneur" |  | ||||||
|                  |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|    |  | ||||||
|   /admin/appointments/report/{appointmentId}: |  | ||||||
|     put: |  | ||||||
|       summary: enregistrer un rapport du rendez-vous |  | ||||||
|       description: |  | ||||||
|         Generate a PDF file formatted  |  | ||||||
|         from input text and links it  |  | ||||||
|         to the appointement. |  | ||||||
|       tags: |  | ||||||
|         - Admin API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-admin |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: appointmentId |  | ||||||
|           required: true |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 title: |  | ||||||
|                   type: string |  | ||||||
|                 body: |  | ||||||
|                   type: string |  | ||||||
|                 conclusion: |  | ||||||
|                   type: string |  | ||||||
|          |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|                     missing or invalid |  | ||||||
|     post: |  | ||||||
|       summary: modifier un rapport déja éxistant du rendez-vous |  | ||||||
|       description: |  | ||||||
|         Modifies the report file to input  |  | ||||||
|         text and links it to the appointement. |  | ||||||
|       tags: |  | ||||||
|         - Admin API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-admin |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: appointmentId |  | ||||||
|           required: true |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 title: |  | ||||||
|                   type: string |  | ||||||
|                 body: |  | ||||||
|                   type: string |  | ||||||
|                 conclusion: |  | ||||||
|                   type: string |  | ||||||
|          |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|                     missing or invalid |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   /admin/projects/remove/{projectId}: |  | ||||||
|     delete: |  | ||||||
|       summary: supression d'un project |  | ||||||
|       description: |  | ||||||
|         Removes the project |  | ||||||
|         with the inputed id projectId |  | ||||||
|       tags: |  | ||||||
|         - Admin API |  | ||||||
|       security: |  | ||||||
|           - MyINPulse: |  | ||||||
|               - MyINPulse-admin |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: projectId |  | ||||||
|           required: true |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|            |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|       |  | ||||||
|  |  | ||||||
|   /admin/projects/pending: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne la liste des projets en attente de validation |  | ||||||
|       tags:  |  | ||||||
|         - Admin API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|       description: |  | ||||||
|         JSON array of who's elements are objects containing |  | ||||||
|         necessary information for the view (project name, |  | ||||||
|         entrepreneur names, etc..) of all pending projects. |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/json: |  | ||||||
|               schema: |  | ||||||
|                 type: array |  | ||||||
|                 items: |  | ||||||
|                   type: object |  | ||||||
|                   properties: |  | ||||||
|                     name: |  | ||||||
|                       type: string |  | ||||||
|                     founder: |  | ||||||
|                       $ref: "#/components/schemas/user-entrepreneur" |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is missing or invalid |  | ||||||
|  |  | ||||||
| # |  | ||||||
| #       ____  _                        _      _    ____ ___  |  | ||||||
| #      / ___|| |__   __ _ _ __ ___  __| |    / \  |  _ \_ _| |  | ||||||
| #      \___ \| '_ \ / _` | '__/ _ \/ _` |   / _ \ | |_) | |  |  | ||||||
| #       ___) | | | | (_| | | |  __/ (_| |  / ___ \|  __/| |  |  | ||||||
| #      |____/|_| |_|\__,_|_|  \___|\__,_| /_/   \_\_|  |___| |  | ||||||
| #                    |  | ||||||
|  |  | ||||||
|   /shared/appointments/upcoming: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne la list des prochains rendez-vous de l'utilisateur |  | ||||||
|       tags:  |  | ||||||
|         - Shared API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|             - MyINPulse-entrepreneur |  | ||||||
|       description: |  | ||||||
|         JSON array of upcoming appointment data (name, date, time etc..) for a user. |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/json: |  | ||||||
|               schema: |  | ||||||
|                 type: array |  | ||||||
|                 items: |  | ||||||
|                   type: object |  | ||||||
|                   properties: |  | ||||||
|                     name: |  | ||||||
|                       type: string |  | ||||||
|                     date: |  | ||||||
|                       type: string |  | ||||||
|                     time: |  | ||||||
|                       type: string |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is missing or invalid         |  | ||||||
|  |  | ||||||
|   /shared/projects/lcsection/{projectId}/{title}/{date}: |  | ||||||
|       get: |  | ||||||
|         summary: Retourne la liste de sections de LC avec un titre donné |  | ||||||
|         tags:  |  | ||||||
|           - Shared API |  | ||||||
|         security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|             - MyINPulse-entrepreneur |  | ||||||
|         description: |  | ||||||
|           JSON array containing Lean Canvas  |  | ||||||
|           section data with a title for the |  | ||||||
|           current date (or given date if the |  | ||||||
|           date parameter is passed) |  | ||||||
|         parameters: |  | ||||||
|           - in: path |  | ||||||
|             required: true |  | ||||||
|             name: projectId |  | ||||||
|             schema: |  | ||||||
|               type: integer |  | ||||||
|           - in: path |  | ||||||
|             required: true |  | ||||||
|             description: this number can be 1, 2,...,8. It is associated with the title of the lcsection |  | ||||||
|             name: title |  | ||||||
|             schema: |  | ||||||
|               type: integer |  | ||||||
|               enum: [1, 2, 3, 4, 5, 6, 7, 8]  |  | ||||||
|              |  | ||||||
|           - in: path |  | ||||||
|             required: true |  | ||||||
|             name: date |  | ||||||
|             description: the date corresponding to the wanted version of lc section. "Nan" for the latest version |  | ||||||
|             example: "NaN" |  | ||||||
|             schema: |  | ||||||
|               type: string |  | ||||||
|                |  | ||||||
|         responses: |  | ||||||
|           "200": |  | ||||||
|             description: OK |  | ||||||
|             content: |  | ||||||
|               application/json: |  | ||||||
|                 schema: |  | ||||||
|                   type: array |  | ||||||
|                   items: |  | ||||||
|                     type: object |  | ||||||
|                     properties: |  | ||||||
|                       section: |  | ||||||
|                         type: string |  | ||||||
|                       txt: |  | ||||||
|                         type: string |  | ||||||
|                       time: |  | ||||||
|                         type: string |  | ||||||
|           "400": |  | ||||||
|             description: Bad request |  | ||||||
|           "401": |  | ||||||
|             description: Authorization information is missing or invalid |  | ||||||
|    |  | ||||||
|   /shared/projects/entrepreneurs/{projectId}: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne la liste d'entrepreneurs associée à un projet donné |  | ||||||
|       tags:  |  | ||||||
|           - Shared API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|             - MyINPulse-entrepreneur |  | ||||||
|       description: |  | ||||||
|         JSON array of entrepreneur |  | ||||||
|         names associated with a project |  | ||||||
|       parameters: |  | ||||||
|           - in: path |  | ||||||
|             name: projectId |  | ||||||
|             schema: |  | ||||||
|               type: integer |  | ||||||
|             required: true |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/json: |  | ||||||
|               schema: |  | ||||||
|                 type: array |  | ||||||
|                 items: |  | ||||||
|                   $ref: "#/components/schemas/user-entrepreneur" |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is missing or invalid |  | ||||||
|    |  | ||||||
|   /shared/projects/admin/{projectId}: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne les informations de l'admin qui accompagne le projet |  | ||||||
|       tags:  |  | ||||||
|           - Shared API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|             - MyINPulse-entrepreneur |  | ||||||
|       description:  |  | ||||||
|         JSON object containing information (name, gmail, tel, etc..)  |  | ||||||
|         the admin supervising the project with id projectID. |  | ||||||
|       parameters: |  | ||||||
|             - in: path |  | ||||||
|               name: projectId |  | ||||||
|               schema: |  | ||||||
|                 type: integer |  | ||||||
|               required: true |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/json: |  | ||||||
|               schema: |  | ||||||
|                 $ref: "#/components/schemas/user-admin" |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|    |  | ||||||
|   /shared/projects/appointments/{projectId}: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne les rendez-vous du projet  |  | ||||||
|       tags:  |  | ||||||
|           - Shared API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|             - MyINPulse-entrepreneur |  | ||||||
|       description:  |  | ||||||
|         JSON array of upcoming and past appointment  |  | ||||||
|         data for the project with id projectID. |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: projectId |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|           required: true |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/json: |  | ||||||
|               schema: |  | ||||||
|                 type: array |  | ||||||
|                 items: |  | ||||||
|                   type: object |  | ||||||
|                   properties: |  | ||||||
|                     appointementId: |  | ||||||
|                       type: integer |  | ||||||
|                     name:  |  | ||||||
|                       type: string |  | ||||||
|                     date:  |  | ||||||
|                       type: string |  | ||||||
|                     time:  |  | ||||||
|                       type: string |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|   /shared/projects/appointments/report/{apointementId}: |  | ||||||
|     get: |  | ||||||
|       summary: Retourne le rapport pdf du rendez-vous |  | ||||||
|       tags:  |  | ||||||
|           - Shared API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|             - MyINPulse-admin |  | ||||||
|             - MyINPulse-entrepreneur |  | ||||||
|       description: |  | ||||||
|         PDF file containing the ap- |  | ||||||
|         pointment report |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: apointementId |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|           required: true |  | ||||||
|       responses: |  | ||||||
|         "200": |  | ||||||
|           description: OK |  | ||||||
|           content: |  | ||||||
|             application/pdf: |  | ||||||
|               schema: |  | ||||||
|                 type: string |  | ||||||
|                 format: binary |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|    |  | ||||||
|   /shared/appointments/request: |  | ||||||
|     post: |  | ||||||
|       summary: demander un rendez-vous |  | ||||||
|       description: |  | ||||||
|         will add an appointement request request by the applicant  |  | ||||||
|         to have an appointment to be confirmed or denied by the |  | ||||||
|         specified participants of the appointement. |  | ||||||
|       tags: |  | ||||||
|         - Shared API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-entrepreneur |  | ||||||
|           - MyINPulse-admin |  | ||||||
|       requestBody: |  | ||||||
|         description: \"participants\" property is an array containing userids of the participants in the appointement |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 title: |  | ||||||
|                   type: string |  | ||||||
|                 start_time: |  | ||||||
|                   type: string |  | ||||||
|                 end_time: |  | ||||||
|                   type: string |  | ||||||
|                 place: |  | ||||||
|                   type: string |  | ||||||
|                 applicantId: |  | ||||||
|                   type: integer |  | ||||||
|                 participants: |  | ||||||
|                   #/* */ |  | ||||||
|                   type: array |  | ||||||
|                   items: |  | ||||||
|                     type: integer |  | ||||||
|                      |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|                     missing or invalid         |  | ||||||
|    |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #      _____ _   _ _____ ____  _____ ____  ____  _____ _   _ _____ _   _ ____   |  | ||||||
| #     | ____| \ | |_   _|  _ \| ____|  _ \|  _ \| ____| \ | | ____| | | |  _ \  |  | ||||||
| #     |  _| |  \| | | | | |_) |  _| | |_) | |_) |  _| |  \| |  _| | | | | |_) | |  | ||||||
| #     | |___| |\  | | | |  _ <| |___|  __/|  _ <| |___| |\  | |___| |_| |  _ <  |  | ||||||
| #     |_____|_|_\_| |_| |_| \_\_____|_|   |_| \_\_____|_| \_|_____|\___/|_| \_\ |  | ||||||
| #        / \  |  _ \_ _|                                                        |  | ||||||
| #       / _ \ | |_) | |                                                         |  | ||||||
| #      / ___ \|  __/| |                                                         |  | ||||||
| #     /_/   \_\_|  |___|                                                        |  | ||||||
| # |  | ||||||
|  |  | ||||||
|                      |  | ||||||
|   /entrepreneur/projects/request: |  | ||||||
|     post: |  | ||||||
|       summary: demander la création et validation d'un projet |  | ||||||
|       tags: |  | ||||||
|         - Entrepreneurs API |  | ||||||
|       description:  |  | ||||||
|         Adds project to pending projects  |  | ||||||
|         to then be accepted or rejected by |  | ||||||
|         an admin |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-entrepreneur |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 name: |  | ||||||
|                   type: string |  | ||||||
|                 founder: |  | ||||||
|                   $ref: "#/components/schemas/user-entrepreneur" |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|              |  | ||||||
|             |  | ||||||
|   /entrepreneur/lcsection/add/{projectId}: |  | ||||||
|     post: |  | ||||||
|       summary: ajouter une sections au LC |  | ||||||
|       description:  |  | ||||||
|         Adds input data to the user's LC  |  | ||||||
|         with a specified title. |  | ||||||
|       tags: |  | ||||||
|        - Entrepreneurs API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-entrepreneur |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: projectId |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|           required: true |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 title: |  | ||||||
|                   type: string |  | ||||||
|                 txt: |  | ||||||
|                   type: string |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|   /entrepreneur/lcsection/modify/{sectionId}: |  | ||||||
|     put: |  | ||||||
|       summary: modifier les données d'une section LC |  | ||||||
|       description:  |  | ||||||
|         Modifies input Lean Canvas section by changing it to  |  | ||||||
|         the information in the request body and changes the |  | ||||||
|         time stamp. |  | ||||||
|       tags: |  | ||||||
|        - Entrepreneurs API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-entrepreneur |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: sectionId |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|           required: true |  | ||||||
|       requestBody: |  | ||||||
|         required: true |  | ||||||
|         content: |  | ||||||
|           application/json: |  | ||||||
|             schema: |  | ||||||
|               type: object |  | ||||||
|               properties: |  | ||||||
|                 title: |  | ||||||
|                   type: string |  | ||||||
|                 txt: |  | ||||||
|                   type: string |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|             missing or invalid |  | ||||||
|              |  | ||||||
|   /entrepreneur/lcsection/remove/{sectionId}: |  | ||||||
|     delete: |  | ||||||
|       summary: supprimer une section LC. |  | ||||||
|       description: |  | ||||||
|         Deletes section from Lean Canvas |  | ||||||
|       tags: |  | ||||||
|         - Entrepreneurs API |  | ||||||
|       security: |  | ||||||
|         - MyINPulse: |  | ||||||
|           - MyINPulse-entrepreneur |  | ||||||
|       parameters: |  | ||||||
|         - in: path |  | ||||||
|           name: sectionId |  | ||||||
|           schema: |  | ||||||
|             type: integer |  | ||||||
|           required: true |  | ||||||
|       responses: |  | ||||||
|         "200":  |  | ||||||
|           description: OK |  | ||||||
|         "400": |  | ||||||
|           description: Bad request |  | ||||||
|         "401": |  | ||||||
|           description: Authorization information is |  | ||||||
|                       missing or invalid |  | ||||||
|    |  | ||||||
|    |  | ||||||
|    |  | ||||||
|    |  | ||||||
							
								
								
									
										60
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										60
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,16 +1,13 @@ | |||||||
| help: | help: | ||||||
| 	@echo "make [clean dev-front prod dev-back dev]" | 	@echo "make [clean dev-front prod dev-back]" | ||||||
|  |  | ||||||
| clean:  | clean:  | ||||||
| 	@cp config/frontdev.env front/MyINPulse-front/.env |  | ||||||
| 	@cp config/frontdev.env .env |  | ||||||
| 	@cp config/frontdev.env MyINPulse-back/.env |  | ||||||
| 	@cp config/prod.docker-compose.yaml docker-compose.yaml | 	@cp config/prod.docker-compose.yaml docker-compose.yaml | ||||||
| 	@docker compose down | 	@docker compose down | ||||||
| 	@rm -f docker-compose.yaml | 	@rm -f docker-compose.yaml | ||||||
| 	@rm -f .env | 	@rm -f .env | ||||||
| 	@rm -f front/MyINPulse-front/.env | 	@rm -f front/MyINPulse-front/.env | ||||||
| 	@rm -f MyINPulse-back/.env |  | ||||||
|  |  | ||||||
| # Install npm packages | # Install npm packages | ||||||
| front/MyINPulse-front/.installed: | front/MyINPulse-front/.installed: | ||||||
| @@ -19,55 +16,26 @@ front/MyINPulse-front/.installed: | |||||||
|  |  | ||||||
| vite: ./front/MyINPulse-front/.installed | vite: ./front/MyINPulse-front/.installed | ||||||
|  |  | ||||||
| keycloak: ./keycloak/.installed |  | ||||||
|  |  | ||||||
| keycloak/.installed: | dev-front: clean vite | ||||||
| 	@echo "running one time install" | 	@cp config/frontdev.front.env front/MyINPulse-front/.env | ||||||
| 	@cd keycloak/CAS && sudo sh build.sh | 	@cp config/frontdev.main.env .env | ||||||
| 	@touch ./keycloak/.installed |  | ||||||
|  |  | ||||||
| dev-front: clean vite keycloak |  | ||||||
| 	@cp config/frontdev.env front/MyINPulse-front/.env |  | ||||||
| 	@cp config/frontdev.env .env |  | ||||||
| 	@cp config/frontdev.env MyINPulse-back/.env |  | ||||||
| 	@cp config/frontdev.docker-compose.yaml docker-compose.yaml | 	@cp config/frontdev.docker-compose.yaml docker-compose.yaml | ||||||
| 	@docker compose up -d --build | 	@docker compose up -d --build | ||||||
| 	@cd ./front/MyINPulse-front/ && npm run dev | 	@cd ./front/MyINPulse-front/ && npm run dev | ||||||
|  |  | ||||||
| prod: clean keycloak | prod: clean | ||||||
| 	@cp config/prod.env front/MyINPulse-front/.env | 	@cp config/prod.front.env front/MyINPulse-front/.env | ||||||
| 	@cp config/prod.env .env | 	@cp config/prod.main.env .env | ||||||
| 	@cp config/prod.env .env | 	@cp config/frontdev.docker-compose.yaml docker-compose.yaml | ||||||
| 	@cp config/prod.docker-compose.yaml docker-compose.yaml |  | ||||||
| 	@docker compose up -d --build | 	@docker compose up -d --build | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| dev-back: keycloak | dev-back: | ||||||
| 	@cp config/backdev.env front/MyINPulse-front/.env | 	@cp config/backdev.front.env front/MyINPulse-front/.env | ||||||
| 	@cp config/backdev.env .env | 	@cp config/backdev.main.env .env | ||||||
| 	@cp config/backdev.env MyINPulse-back/.env |  | ||||||
| 	@cp config/backdev.docker-compose.yaml docker-compose.yaml | 	@cp config/backdev.docker-compose.yaml docker-compose.yaml | ||||||
| 	@docker compose up -d --build | 	@docker compose up -d --build | ||||||
| 	@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)' | 	@echo "cd MyINPulse-back" | ||||||
| 	@echo "./gradlew bootRun --args='--server.port=8081'" | 	@echo "./gradlew bootRun --args='--server.port=8081'" | ||||||
| 	 |  | ||||||
| dev: clean vite keycloak |  | ||||||
| 	@cp config/dev.env front/MyINPulse-front/.env |  | ||||||
| 	@cp config/dev.env .env |  | ||||||
| 	@cp config/dev.env MyINPulse-back/.env |  | ||||||
| 	@cp config/dev.docker-compose.yaml docker-compose.yaml |  | ||||||
| 	@docker compose up -d --build |  | ||||||
| 	@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)' |  | ||||||
| 	@echo "./gradlew bootRun --args='--server.port=8081'" |  | ||||||
| 	@cd ./front/MyINPulse-front/ && npm run dev & |  | ||||||
|  |  | ||||||
| test-back: clean keycloak |  | ||||||
| 	@cp config/dev.env front/MyINPulse-front/.env |  | ||||||
| 	@cp config/dev.env .env |  | ||||||
| 	@cp config/dev.env MyINPulse-back/.env |  | ||||||
| 	@cp config/dev.docker-compose.yaml docker-compose.yaml |  | ||||||
| 	@docker compose up -d --build |  | ||||||
| 	@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)' |  | ||||||
| 	@cd ./MyINPulse-back/ && ./gradlew test && ./gradlew jacocoTestReport |  | ||||||
| 	@firefox ./MyINPulse-back/build/jacocoHtml/index.html |  | ||||||
| @@ -1,59 +1,29 @@ | |||||||
| plugins { | plugins { | ||||||
|     id 'java' | 	id 'java' | ||||||
|     id 'org.springframework.boot' version '3.4.2' | 	id 'org.springframework.boot' version '3.4.2' | ||||||
|     id 'io.spring.dependency-management' version '1.1.7' | 	id 'io.spring.dependency-management' version '1.1.7' | ||||||
|     id 'jacoco' |  | ||||||
| } | } | ||||||
|  |  | ||||||
| group = 'enseirb' | group = 'enseirb' | ||||||
| version = '0.0.1-SNAPSHOT' | version = '0.0.1-SNAPSHOT' | ||||||
|  |  | ||||||
| java { | java { | ||||||
|     toolchain { | 	toolchain { | ||||||
|         languageVersion = JavaLanguageVersion.of(21) | 		languageVersion = JavaLanguageVersion.of(21) | ||||||
|     } | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| repositories { | repositories { | ||||||
|     mavenCentral() | 	mavenCentral() | ||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-oauth2-resource-server' | 	implementation 'org.springframework.boot:spring-boot-starter-oauth2-resource-server' | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-web' | 	implementation 'org.springframework.boot:spring-boot-starter-web' | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-data-jpa' | 	testImplementation 'org.springframework.boot:spring-boot-starter-test' | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-validation' | 	testRuntimeOnly 'org.junit.platform:junit-platform-launcher' | ||||||
|     implementation 'org.springframework.boot:spring-boot-starter-data-rest' |  | ||||||
|     implementation group: 'org.apache.logging.log4j', name: 'log4j-api', version: '2.+' |  | ||||||
|     implementation group: 'org.apache.logging.log4j', name: 'log4j-core', version: '2.+' |  | ||||||
|     implementation 'org.postgresql:postgresql' |  | ||||||
|     implementation group: 'com.itextpdf', name: 'itextpdf', version: '5.5.13.3' |  | ||||||
|  |  | ||||||
|     testImplementation 'org.springframework.boot:spring-boot-starter-test' |  | ||||||
|     testImplementation 'com.h2database:h2' |  | ||||||
|  |  | ||||||
|     testRuntimeOnly 'org.junit.platform:junit-platform-launcher' |  | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks.named('test') { | tasks.named('test') { | ||||||
|     useJUnitPlatform() | 	useJUnitPlatform() | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| test { |  | ||||||
|     finalizedBy jacocoTestReport // report is always generated after tests run |  | ||||||
| } |  | ||||||
| jacocoTestReport { |  | ||||||
|     dependsOn test // tests are required to run before generating the report |  | ||||||
|     reports { |  | ||||||
|         xml.required = false |  | ||||||
|         csv.required = false |  | ||||||
|         html.outputLocation = layout.buildDirectory.dir('jacocoHtml') |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| jacoco { |  | ||||||
|     toolVersion = "0.8.12" |  | ||||||
|     reportsDirectory = layout.buildDirectory.dir('customJacocoReportDir') |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,15 +1,29 @@ | |||||||
| package enseirb.myinpulse; | package enseirb.myinpulse; | ||||||
|  |  | ||||||
|  | import enseirb.myinpulse.security.KeycloakJwtRolesConverter; | ||||||
| import org.springframework.boot.SpringApplication; | import org.springframework.boot.SpringApplication; | ||||||
| import org.springframework.boot.autoconfigure.SpringBootApplication; | import org.springframework.boot.autoconfigure.SpringBootApplication; | ||||||
|  | import org.springframework.context.annotation.Bean; | ||||||
|  | import org.springframework.core.convert.converter.Converter; | ||||||
|  | import org.springframework.security.authentication.AbstractAuthenticationToken; | ||||||
|  | import org.springframework.security.config.annotation.web.builders.HttpSecurity; | ||||||
| import org.springframework.security.oauth2.jwt.*; | import org.springframework.security.oauth2.jwt.*; | ||||||
|  | import org.springframework.security.web.SecurityFilterChain; | ||||||
|  | import org.springframework.web.cors.CorsConfiguration; | ||||||
|  | import org.springframework.web.cors.CorsConfigurationSource; | ||||||
|  | import org.springframework.web.cors.UrlBasedCorsConfigurationSource; | ||||||
|  |  | ||||||
| import java.util.*; | import java.util.*; | ||||||
|  | import java.util.stream.Collectors; | ||||||
|  |  | ||||||
|  | import static org.springframework.security.authorization.AuthorityAuthorizationManager.hasRole; | ||||||
|  |  | ||||||
| @SpringBootApplication | @SpringBootApplication | ||||||
| public class MyinpulseApplication { | public class MyinpulseApplication { | ||||||
|  |  | ||||||
|     public static void main(String[] args) { | 	public static void main(String[] args) { | ||||||
|         SpringApplication.run(MyinpulseApplication.class, args); | 		SpringApplication.run(MyinpulseApplication.class, args); | ||||||
|     } | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | package enseirb.myinpulse.api; | ||||||
|  |  | ||||||
|  | import org.springframework.boot.autoconfigure.SpringBootApplication; | ||||||
|  | import org.springframework.security.core.context.SecurityContextHolder; | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping; | ||||||
|  | import org.springframework.web.bind.annotation.RequestMethod; | ||||||
|  | import org.springframework.web.bind.annotation.RestController; | ||||||
|  | import org.springframework.web.bind.annotation.CrossOrigin; | ||||||
|  |  | ||||||
|  | import java.security.Principal; | ||||||
|  |  | ||||||
|  | @SpringBootApplication | ||||||
|  | @RestController | ||||||
|  | public class GetUserInfo { | ||||||
|  |     // TODO: understand how to get data | ||||||
|  |     @GetMapping("/getUserInfo") | ||||||
|  |     public Object user(Principal principal) { | ||||||
|  |         System.out.println("GetUserInfo + " + principal); | ||||||
|  |         System.out.println(SecurityContextHolder.getContext().getAuthentication()); | ||||||
|  |         return SecurityContextHolder.getContext().getAuthentication().getPrincipal(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @CrossOrigin(methods = {RequestMethod.GET, RequestMethod.OPTIONS}) | ||||||
|  |     @GetMapping("/random") | ||||||
|  |     public boolean rand(){ | ||||||
|  |         System.err.println("HELLO"); | ||||||
|  |         return Math.random() > 0.5; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @CrossOrigin(methods = {RequestMethod.GET, RequestMethod.OPTIONS}) | ||||||
|  |     @GetMapping("/random2") | ||||||
|  |     public boolean rand2(){ | ||||||
|  |         System.err.println("HELLO2"); | ||||||
|  |         return Math.random() > 0.5; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @CrossOrigin(methods = {RequestMethod.GET, RequestMethod.OPTIONS}) | ||||||
|  |     @GetMapping("/random3") | ||||||
|  |     public boolean rand3(){ | ||||||
|  |         System.err.println("HELLO"); | ||||||
|  |         return Math.random() > 0.5; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,107 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Source: https://github.com/ChristianHuff-DEV/secure-spring-rest-api-using-keycloak/blob/main/src/main/java/io/betweendata/RestApi/security/oauth2/KeycloakJwtRolesConverter.java |  | ||||||
|  * edited by Pierre Tellier |  | ||||||
|  */ |  | ||||||
| package enseirb.myinpulse.config; |  | ||||||
|  |  | ||||||
| import static java.util.stream.Collectors.toSet; |  | ||||||
|  |  | ||||||
| import org.springframework.core.convert.converter.Converter; |  | ||||||
| import org.springframework.security.authentication.AbstractAuthenticationToken; |  | ||||||
| import org.springframework.security.core.GrantedAuthority; |  | ||||||
| import org.springframework.security.core.authority.SimpleGrantedAuthority; |  | ||||||
| import org.springframework.security.oauth2.jwt.Jwt; |  | ||||||
| import org.springframework.security.oauth2.server.resource.authentication.JwtAuthenticationToken; |  | ||||||
| import org.springframework.security.oauth2.server.resource.authentication.JwtGrantedAuthoritiesConverter; |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.Collection; |  | ||||||
| import java.util.Map; |  | ||||||
| import java.util.stream.Collectors; |  | ||||||
| import java.util.stream.Stream; |  | ||||||
|  |  | ||||||
| public class KeycloakJwtRolesConverter implements Converter<Jwt, AbstractAuthenticationToken> { |  | ||||||
|     /** Prefix used for realm level roles. */ |  | ||||||
|     public static final String PREFIX_REALM_ROLE = "ROLE_REALM_"; |  | ||||||
|  |  | ||||||
|     /** Prefix used in combination with the resource (client) name for resource level roles. */ |  | ||||||
|     public static final String PREFIX_RESOURCE_ROLE = "ROLE_"; |  | ||||||
|  |  | ||||||
|     /** Name of the claim containing the realm level roles */ |  | ||||||
|     private static final String CLAIM_REALM_ACCESS = "realm_access"; |  | ||||||
|  |  | ||||||
|     /** Name of the claim containing the resources (clients) the user has access to. */ |  | ||||||
|     private static final String CLAIM_RESOURCE_ACCESS = "resource_access"; |  | ||||||
|  |  | ||||||
|     /** Name of the claim containing roles. (Applicable to realm and resource level.) */ |  | ||||||
|     private static final String CLAIM_ROLES = "roles"; |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public AbstractAuthenticationToken convert(Jwt source) { |  | ||||||
|         return new JwtAuthenticationToken( |  | ||||||
|                 source, |  | ||||||
|                 Stream.concat( |  | ||||||
|                                 new JwtGrantedAuthoritiesConverter().convert(source).stream(), |  | ||||||
|                                 tokenRolesExtractor(source).stream()) |  | ||||||
|                         .collect(toSet())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Extracts the realm and resource level roles from a JWT token distinguishing between them |  | ||||||
|      * using prefixes. |  | ||||||
|      */ |  | ||||||
|     public Collection<GrantedAuthority> tokenRolesExtractor(Jwt jwt) { |  | ||||||
|         // Collection that will hold the extracted roles |  | ||||||
|         Collection<GrantedAuthority> grantedAuthorities = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|         // Realm roles |  | ||||||
|         // Get the part of the access token that holds the roles assigned on realm level |  | ||||||
|         Map<String, Collection<String>> realmAccess = jwt.getClaim(CLAIM_REALM_ACCESS); |  | ||||||
|  |  | ||||||
|         // Verify that the claim exists and is not empty |  | ||||||
|         if (realmAccess != null && !realmAccess.isEmpty()) { |  | ||||||
|             // From the realm_access claim get the roles |  | ||||||
|             Collection<String> roles = realmAccess.get(CLAIM_ROLES); |  | ||||||
|             // Check if any roles are present |  | ||||||
|             if (roles != null && !roles.isEmpty()) { |  | ||||||
|                 // Iterate of the roles and add them to the granted authorities |  | ||||||
|                 Collection<GrantedAuthority> realmRoles = |  | ||||||
|                         roles.stream() |  | ||||||
|                                 // Prefix all realm roles with "ROLE_realm_" |  | ||||||
|                                 .map(role -> new SimpleGrantedAuthority(PREFIX_REALM_ROLE + role)) |  | ||||||
|                                 .collect(Collectors.toList()); |  | ||||||
|                 grantedAuthorities.addAll(realmRoles); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Resource (client) roles |  | ||||||
|         // A user might have access to multiple resources all containing their own roles. Therefore, |  | ||||||
|         // it is a map of |  | ||||||
|         // resource each possibly containing a "roles" property. |  | ||||||
|         Map<String, Map<String, Collection<String>>> resourceAccess = |  | ||||||
|                 jwt.getClaim(CLAIM_RESOURCE_ACCESS); |  | ||||||
|  |  | ||||||
|         // Check if resources are assigned |  | ||||||
|         if (resourceAccess != null && !resourceAccess.isEmpty()) { |  | ||||||
|             // Iterate of all the resources |  | ||||||
|             resourceAccess.forEach( |  | ||||||
|                     (resource, resourceClaims) -> { |  | ||||||
|                         // Iterate of the "roles" claim inside the resource claims |  | ||||||
|                         resourceClaims |  | ||||||
|                                 .get(CLAIM_ROLES) |  | ||||||
|                                 .forEach( |  | ||||||
|                                         // Add the role to the granted authority prefixed with ROLE_ |  | ||||||
|                                         // and the name of the resource |  | ||||||
|                                         role -> |  | ||||||
|                                                 grantedAuthorities.add( |  | ||||||
|                                                         new SimpleGrantedAuthority( |  | ||||||
|                                                                 PREFIX_RESOURCE_ROLE |  | ||||||
|                                                                         + resource |  | ||||||
|                                                                         + "_" |  | ||||||
|                                                                         + role))); |  | ||||||
|                     }); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return grantedAuthorities; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,8 +1,6 @@ | |||||||
| package enseirb.myinpulse.config; | package enseirb.myinpulse.config; | ||||||
|  |  | ||||||
| import static org.springframework.security.authorization.AuthorityAuthorizationManager.hasRole; | import enseirb.myinpulse.security.KeycloakJwtRolesConverter; | ||||||
|  |  | ||||||
| import org.springframework.beans.factory.annotation.Value; |  | ||||||
| import org.springframework.context.annotation.Bean; | import org.springframework.context.annotation.Bean; | ||||||
| import org.springframework.context.annotation.Configuration; | import org.springframework.context.annotation.Configuration; | ||||||
| import org.springframework.security.config.annotation.web.builders.HttpSecurity; | import org.springframework.security.config.annotation.web.builders.HttpSecurity; | ||||||
| @@ -14,62 +12,39 @@ import org.springframework.web.cors.UrlBasedCorsConfigurationSource; | |||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  |  | ||||||
|  | import static org.springframework.security.authorization.AuthorityAuthorizationManager.hasRole; | ||||||
|  |  | ||||||
| @Configuration | @Configuration | ||||||
| public class WebSecurityCustomConfiguration { | public class WebSecurityCustomConfiguration { | ||||||
|     // CORS configuration |     // CORS configuration | ||||||
|  |     // TODO: make sure to only accept our own domains | ||||||
|     @Value("${VITE_APP_URL}") |  | ||||||
|     private String frontendUrl; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Configure the CORS (Cross Origin Ressource Sharing -- a security feature) configuration. The |  | ||||||
|      * only allowed website is the frontend, defined in the .env file. |  | ||||||
|      * |  | ||||||
|      * @return the CORS configuration used by the backend |  | ||||||
|      */ |  | ||||||
|     @Bean |     @Bean | ||||||
|     public CorsConfigurationSource corsConfigurationSource() { |     public CorsConfigurationSource corsConfigurationSource() { | ||||||
|         CorsConfiguration configuration = new CorsConfiguration(); |         CorsConfiguration configuration = new CorsConfiguration(); | ||||||
|         configuration.setAllowedOrigins(List.of(frontendUrl)); |         configuration.setAllowedOrigins(List.of("*")); | ||||||
|         configuration.setAllowedMethods(Arrays.asList("GET", "OPTIONS")); |         configuration.setAllowedMethods(Arrays.asList("GET", "OPTIONS")); | ||||||
|         configuration.setAllowedHeaders( |         configuration.setAllowedHeaders(Arrays.asList("authorization", "content-type", | ||||||
|                 Arrays.asList("authorization", "content-type", "x-auth-token")); |                 "x-auth-token")); // Do not remove, this fixes the CORS errors when unauthenticated | ||||||
|         UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource(); |         UrlBasedCorsConfigurationSource source = new | ||||||
|  |                 UrlBasedCorsConfigurationSource(); | ||||||
|         source.registerCorsConfiguration("/**", configuration); |         source.registerCorsConfiguration("/**", configuration); | ||||||
|  |  | ||||||
|         return source; |         return source; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Configure the authorisation required for each path. |  | ||||||
|      * |  | ||||||
|      * <p>admin endpoints are under /admin/* and entrepreneur are under /entrepreneur/* |  | ||||||
|      * |  | ||||||
|      * <p>If endpoints dont require authentication, they are under /unauth/ |  | ||||||
|      * |  | ||||||
|      * @param http automatically filled in by spring. |  | ||||||
|      * @return a securityfilterchain, automatically used by spring. |  | ||||||
|      * @throws Exception TODO: figure out when the exception are raised |  | ||||||
|      */ |  | ||||||
|     @Bean |     @Bean | ||||||
|     public SecurityFilterChain filterChain(HttpSecurity http) throws Exception { |     public SecurityFilterChain filterChain(HttpSecurity http) throws Exception { | ||||||
|         http.authorizeHttpRequests( |         http | ||||||
|                         authorize -> |                 .authorizeHttpRequests(authorize -> authorize | ||||||
|                                 authorize |                         .requestMatchers("/random2").access(hasRole("REALM_MyINPulse-entrepreneur")) | ||||||
|                                         .requestMatchers("/entrepreneur/**", "/shared/**") |                         .requestMatchers("/random").access(hasRole("REALM_MyINPulse-admin")) | ||||||
|                                         .access(hasRole("REALM_MyINPulse-entrepreneur")) |                         .requestMatchers("/random3").permitAll() | ||||||
|                                         .requestMatchers("/admin/**", "/shared/**") |                         .anyRequest().authenticated() | ||||||
|                                         .access(hasRole("REALM_MyINPulse-admin")) |                 ) | ||||||
|                                         .requestMatchers("/unauth/**") |                 .oauth2ResourceServer(oauth2 -> oauth2 | ||||||
|                                         .permitAll() |                         .jwt(jwt -> jwt. | ||||||
|                                         .anyRequest() |                                 jwtAuthenticationConverter(new KeycloakJwtRolesConverter()))); | ||||||
|                                         .authenticated()) |  | ||||||
|                 .oauth2ResourceServer( |  | ||||||
|                         oauth2 -> |  | ||||||
|                                 oauth2.jwt( |  | ||||||
|                                         jwt -> |  | ||||||
|                                                 jwt.jwtAuthenticationConverter( |  | ||||||
|                                                         new KeycloakJwtRolesConverter()))); |  | ||||||
|         return http.build(); |         return http.build(); | ||||||
|  |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,102 +0,0 @@ | |||||||
| package enseirb.myinpulse.controller; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.*; |  | ||||||
| import enseirb.myinpulse.service.AdminApiService; |  | ||||||
|  |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.boot.autoconfigure.SpringBootApplication; |  | ||||||
| import org.springframework.security.core.annotation.AuthenticationPrincipal; |  | ||||||
| import org.springframework.security.oauth2.jwt.Jwt; |  | ||||||
| import org.springframework.web.bind.annotation.*; |  | ||||||
|  |  | ||||||
| @SpringBootApplication |  | ||||||
| @RestController |  | ||||||
| public class AdminApi { |  | ||||||
|  |  | ||||||
|     private final AdminApiService adminApiService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     AdminApi(AdminApiService adminApiService) { |  | ||||||
|         this.adminApiService = adminApiService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: description |  | ||||||
|      * |  | ||||||
|      * @return a list of all project managed by the current admin user |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/admin/projects") |  | ||||||
|     public Iterable<Project> getProjects(@AuthenticationPrincipal Jwt principal) { |  | ||||||
|         return adminApiService.getProjectsOfAdmin(principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: Why in admin instead of shared ? + desc |  | ||||||
|      * |  | ||||||
|      * @return a list of upcoming appointments for the current user |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/admin/appointments/upcoming") |  | ||||||
|     public Iterable<Appointment> getUpcomingAppointments(@AuthenticationPrincipal Jwt principal) { |  | ||||||
|         return adminApiService.getUpcomingAppointments(principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: description |  | ||||||
|      * |  | ||||||
|      * @return a list of current unvalidated projects, waiting to be accepted |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/admin/projects/pending") |  | ||||||
|     public Iterable<Project> getPendingProjects() { |  | ||||||
|         return adminApiService.getPendingProjects(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to make a decision about a project. |  | ||||||
|      * |  | ||||||
|      * <p>The decision must contain the administrator |  | ||||||
|      * |  | ||||||
|      * @return the status code of the request |  | ||||||
|      */ |  | ||||||
|     @PostMapping("/admin/projects/decision") |  | ||||||
|     public void validateProject(@RequestBody ProjectDecision decision) { |  | ||||||
|         adminApiService.validateProject(decision); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to manually add a project by an admin |  | ||||||
|      * |  | ||||||
|      * @return the status code of the request |  | ||||||
|      */ |  | ||||||
|     @PostMapping("/admin/project/add") |  | ||||||
|     public void addNewProject(@RequestBody Project project) { |  | ||||||
|         adminApiService.addNewProject(project); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: shouldn't it be an PUT request ? / What is the rerun type |  | ||||||
|      * |  | ||||||
|      * <p>Endpoint used to add a new report to an appointment |  | ||||||
|      * |  | ||||||
|      * @return the status code of the request |  | ||||||
|      */ |  | ||||||
|     @PostMapping("/admin/appoitements/report/{appointmentId}") |  | ||||||
|     public void createAppointmentReport( |  | ||||||
|             @PathVariable long appointmentId, |  | ||||||
|             @RequestBody Report report, |  | ||||||
|             @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         adminApiService.createAppointmentReport( |  | ||||||
|                 appointmentId, report, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: Shouldn't a project be kept in history ? 2 different endpoints ? |  | ||||||
|      * |  | ||||||
|      * <p>Endpoint used to completely remove a project. |  | ||||||
|      * |  | ||||||
|      * @return the status code of the request |  | ||||||
|      */ |  | ||||||
|     @DeleteMapping("/admin/projects/remove/{projectId}") |  | ||||||
|     public void deleteProject(@PathVariable long projectId) { |  | ||||||
|         adminApiService.deleteProject(projectId); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,78 +0,0 @@ | |||||||
| package enseirb.myinpulse.controller; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.SectionCell; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.service.EntrepreneurApiService; |  | ||||||
|  |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.boot.autoconfigure.SpringBootApplication; |  | ||||||
| import org.springframework.security.core.annotation.AuthenticationPrincipal; |  | ||||||
| import org.springframework.security.oauth2.jwt.Jwt; |  | ||||||
| import org.springframework.web.bind.annotation.*; |  | ||||||
|  |  | ||||||
| @SpringBootApplication |  | ||||||
| @RestController |  | ||||||
| public class EntrepreneurApi { |  | ||||||
|  |  | ||||||
|     private final EntrepreneurApiService entrepreneurApiService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     EntrepreneurApi(EntrepreneurApiService entrepreneurApiService) { |  | ||||||
|         this.entrepreneurApiService = entrepreneurApiService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: check return type |  | ||||||
|      * |  | ||||||
|      * <p>Endpoint used to update a LC section. |  | ||||||
|      * |  | ||||||
|      * @return status code |  | ||||||
|      */ |  | ||||||
|     @PutMapping("/entrepreneur/lcsection/modify/{sectionId}") |  | ||||||
|     public void editSectionCell( |  | ||||||
|             @PathVariable Long sectionId, |  | ||||||
|             @RequestBody SectionCell sectionCell, |  | ||||||
|             @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         entrepreneurApiService.editSectionCell( |  | ||||||
|                 sectionId, sectionCell, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: checkReturn Type |  | ||||||
|      * |  | ||||||
|      * <p>Endpoint used to delete a LC section |  | ||||||
|      * |  | ||||||
|      * @return status code |  | ||||||
|      */ |  | ||||||
|     @DeleteMapping("/entrepreneur/lcsection/remove/{sectionId}") |  | ||||||
|     public void removeSectionCell( |  | ||||||
|             @PathVariable Long sectionId, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         entrepreneurApiService.removeSectionCell(sectionId, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: check return type |  | ||||||
|      * |  | ||||||
|      * <p>Endpoint used to create a new LC section |  | ||||||
|      * |  | ||||||
|      * @return status code |  | ||||||
|      */ |  | ||||||
|     @PostMapping("/entrepreneur/lcsection/add") // remove id from doc aswell |  | ||||||
|     public void addLCSection( |  | ||||||
|             @RequestBody SectionCell sectionCell, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         entrepreneurApiService.addSectionCell(sectionCell, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: check return type |  | ||||||
|      * |  | ||||||
|      * <p>Endpoint used to request the creation of a new project |  | ||||||
|      * |  | ||||||
|      * @return status code |  | ||||||
|      */ |  | ||||||
|     @PostMapping("/entrepreneur/project/request") |  | ||||||
|     public void requestNewProject( |  | ||||||
|             @RequestBody Project project, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         entrepreneurApiService.requestNewProject(project, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,105 +0,0 @@ | |||||||
| package enseirb.myinpulse.controller; |  | ||||||
|  |  | ||||||
| import com.itextpdf.text.DocumentException; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.*; |  | ||||||
| import enseirb.myinpulse.service.SharedApiService; |  | ||||||
|  |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.boot.autoconfigure.SpringBootApplication; |  | ||||||
| import org.springframework.security.core.annotation.AuthenticationPrincipal; |  | ||||||
| import org.springframework.security.oauth2.jwt.Jwt; |  | ||||||
| import org.springframework.web.bind.annotation.*; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
|  |  | ||||||
| @SpringBootApplication |  | ||||||
| @RestController |  | ||||||
| public class SharedApi { |  | ||||||
|  |  | ||||||
|     private final SharedApiService sharedApiService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     SharedApi(SharedApiService sharedApiService) { |  | ||||||
|         this.sharedApiService = sharedApiService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to get the data inside the lean canvas |  | ||||||
|      * |  | ||||||
|      * @return a list of lean canvas sections |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/shared/project/lcsection/{projectId}/{sectionId}/{date}") |  | ||||||
|     public Iterable<SectionCell> getLCSection( |  | ||||||
|             @PathVariable("projectId") Long projectId, |  | ||||||
|             @PathVariable("sectionId") Long sectionId, |  | ||||||
|             @PathVariable("date") String date, |  | ||||||
|             @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         return sharedApiService.getSectionCells( |  | ||||||
|                 projectId, sectionId, date, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to get entrepreneurs details |  | ||||||
|      * |  | ||||||
|      * @return a list of all entrepreneurs in a project |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/shared/entrepreneurs/{projectId}") |  | ||||||
|     public Iterable<Entrepreneur> getEntrepreneursByProjectId( |  | ||||||
|             @PathVariable int projectId, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         return sharedApiService.getEntrepreneursByProjectId( |  | ||||||
|                 projectId, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to get the administrator of a project. |  | ||||||
|      * |  | ||||||
|      * @return the admin of a project |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/shared/projects/admin/{projectId}") |  | ||||||
|     public Administrator getAdminByProjectId( |  | ||||||
|             @PathVariable int projectId, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         return sharedApiService.getAdminByProjectId(projectId, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to get all appointments of a single project. |  | ||||||
|      * |  | ||||||
|      * @return a list of all appointments. |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/shared/projects/appointments/{projectId}") |  | ||||||
|     public Iterable<Appointment> getAppointmentsByProjectId( |  | ||||||
|             @PathVariable int projectId, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         return sharedApiService.getAppointmentsByProjectId( |  | ||||||
|                 projectId, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Endpoint used to generate a PDF report |  | ||||||
|      * |  | ||||||
|      * @return a PDF file? TODO: how does that works ? |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/shared/projects/appointments/report/{appointmentId}") |  | ||||||
|     public void getPDFReport( |  | ||||||
|             @PathVariable int appointmentId, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         try { |  | ||||||
|             sharedApiService.getPDFReport(appointmentId, principal.getClaimAsString("email")); |  | ||||||
|         } catch (DocumentException e) { |  | ||||||
|             System.out.println(e + "Document exception"); |  | ||||||
|         } catch (URISyntaxException e) { |  | ||||||
|             System.out.println(e + "Error with URI"); |  | ||||||
|         } catch (IOException e) { |  | ||||||
|             System.out.println(e + "Failed to access file"); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * @return TODO |  | ||||||
|      */ |  | ||||||
|     @PostMapping("/shared/appointment/request") |  | ||||||
|     public void createAppointmentRequest( |  | ||||||
|             @RequestBody Appointment appointment, @AuthenticationPrincipal Jwt principal) { |  | ||||||
|         sharedApiService.createAppointmentRequest(appointment, principal.getClaimAsString("email")); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| package enseirb.myinpulse.exception; |  | ||||||
|  |  | ||||||
| public class RoleNotFoudException extends RuntimeException { |  | ||||||
|     public RoleNotFoudException(String message) { |  | ||||||
|         super(message); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| package enseirb.myinpulse.exception; |  | ||||||
|  |  | ||||||
| public class UserNotFoundException extends RuntimeException { |  | ||||||
|     public UserNotFoundException(String message) { |  | ||||||
|         super(message); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
| import jakarta.persistence.PrimaryKeyJoinColumn; |  | ||||||
| import jakarta.persistence.Table; |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "administrator") |  | ||||||
| @PrimaryKeyJoinColumn(name = "idAdministrator", referencedColumnName = "idUser") |  | ||||||
| public class Administrator extends User { |  | ||||||
|  |  | ||||||
|     @OneToMany(mappedBy = "projectAdministrator", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<Project> listProject = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     /*@OneToMany(mappedBy = "administratorSectionCell", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private List<SectionCell> listSectionCell = new ArrayList<>();*/ |  | ||||||
|     // should now be useless |  | ||||||
|  |  | ||||||
|     @OneToMany(mappedBy = "administratorAnnotation", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<Annotation> listAnnotation = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     /*@OneToMany(mappedBy = "administratorAppointment", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<Appointment> listAppointment = new ArrayList<>();*/ |  | ||||||
|     // should now be useless |  | ||||||
|  |  | ||||||
|     @OneToOne(mappedBy = "administratorAppointment", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private MakeAppointment makeAppointment; |  | ||||||
|  |  | ||||||
|     public Administrator() {} |  | ||||||
|  |  | ||||||
|     public Administrator( |  | ||||||
|             String userSurname, |  | ||||||
|             String username, |  | ||||||
|             String primaryMail, |  | ||||||
|             String secondaryMail, |  | ||||||
|             String phoneNumber) { |  | ||||||
|         super(null, userSurname, username, primaryMail, secondaryMail, phoneNumber); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<Project> getListProject() { |  | ||||||
|         return listProject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateListProject(Project project) { |  | ||||||
|         listProject.add(project); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<Annotation> getListAnnotation() { |  | ||||||
|         return listAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateListAnnotation(Annotation annotation) { |  | ||||||
|         listAnnotation.add(annotation); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public MakeAppointment getMakeAppointment() { |  | ||||||
|         return makeAppointment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setMakeAppointment(MakeAppointment makeAppointment) { |  | ||||||
|         this.makeAppointment = makeAppointment; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,61 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "annotation") |  | ||||||
| public class Annotation { |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idAnnotation; |  | ||||||
|  |  | ||||||
|     private String comment; |  | ||||||
|  |  | ||||||
|     @ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idSectionCell") |  | ||||||
|     private SectionCell sectionCellAnnotation; |  | ||||||
|  |  | ||||||
|     @ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idAdministrator") |  | ||||||
|     private Administrator administratorAnnotation; |  | ||||||
|  |  | ||||||
|     public Annotation() {} |  | ||||||
|  |  | ||||||
|     public Annotation(Long idAnnotation, String commentary) { |  | ||||||
|         this.idAnnotation = idAnnotation; |  | ||||||
|         this.comment = comment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getComment() { |  | ||||||
|         return comment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setComment(String comment) { |  | ||||||
|         this.comment = comment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getIdAnnotation() { |  | ||||||
|         return idAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setIdAnnotation(Long idAnnotation) { |  | ||||||
|         this.idAnnotation = idAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public SectionCell getSectionCellAnnotation() { |  | ||||||
|         return sectionCellAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setSectionCellAnnotation(SectionCell sectionCellAnnotation) { |  | ||||||
|         this.sectionCellAnnotation = sectionCellAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Administrator getAdministratorAnnotation() { |  | ||||||
|         return administratorAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAdministratorAnnotation(Administrator administratorAnnotation) { |  | ||||||
|         this.administratorAnnotation = administratorAnnotation; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,126 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
|  |  | ||||||
| import java.time.LocalDate; |  | ||||||
| import java.time.LocalTime; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "appointment") |  | ||||||
| public class Appointment { |  | ||||||
|  |  | ||||||
|     /*@OneToMany(mappedBy = "appointmentEntrepreneurs", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<Entrepreneur> listEntrepreneur = |  | ||||||
|             new ArrayList<>(); */ |  | ||||||
|     // should now be useless |  | ||||||
|  |  | ||||||
|     @ManyToMany( |  | ||||||
|             fetch = FetchType.LAZY, |  | ||||||
|             cascade = {CascadeType.ALL}) |  | ||||||
|     @JoinTable( |  | ||||||
|             name = "concern", |  | ||||||
|             joinColumns = @JoinColumn(name = "idAppointment"), |  | ||||||
|             inverseJoinColumns = @JoinColumn(name = "idSectionCell")) |  | ||||||
|     List<SectionCell> listSectionCell = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     @OneToOne(mappedBy = "appointmentReport", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private Report report; |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idAppointment; |  | ||||||
|  |  | ||||||
|     private LocalDate appointmentDate; |  | ||||||
|  |  | ||||||
|     private LocalTime appointmentTime; |  | ||||||
|  |  | ||||||
|     private LocalTime appointmentDuration; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String appointmentPlace; |  | ||||||
|  |  | ||||||
|     private String appointmentSubject; |  | ||||||
|  |  | ||||||
|     public Appointment() {} |  | ||||||
|  |  | ||||||
|     public Appointment( |  | ||||||
|             Long idAppointment, |  | ||||||
|             LocalDate appointmentDate, |  | ||||||
|             LocalTime appointmentTime, |  | ||||||
|             LocalTime appointmentDuration, |  | ||||||
|             String appointmentPlace, |  | ||||||
|             String appointmentSubject) { |  | ||||||
|         this.idAppointment = idAppointment; |  | ||||||
|         this.appointmentDate = appointmentDate; |  | ||||||
|         this.appointmentTime = appointmentTime; |  | ||||||
|         this.appointmentDuration = appointmentDuration; |  | ||||||
|         this.appointmentPlace = appointmentPlace; |  | ||||||
|         this.appointmentSubject = appointmentSubject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getIdAppointment() { |  | ||||||
|         return idAppointment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setIdAppointment(Long idAppointment) { |  | ||||||
|         this.idAppointment = idAppointment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public LocalDate getAppointmentDate() { |  | ||||||
|         return appointmentDate; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentDate(LocalDate appointmentDate) { |  | ||||||
|         this.appointmentDate = appointmentDate; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public LocalTime getAppointmentTime() { |  | ||||||
|         return appointmentTime; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentTime(LocalTime appointmentTime) { |  | ||||||
|         this.appointmentTime = appointmentTime; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public LocalTime getAppointmentDuration() { |  | ||||||
|         return appointmentDuration; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentDuration(LocalTime appointmentDuration) { |  | ||||||
|         this.appointmentDuration = appointmentDuration; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getAppointmentPlace() { |  | ||||||
|         return appointmentPlace; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentPlace(String appointmentPlace) { |  | ||||||
|         this.appointmentPlace = appointmentPlace; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getAppointmentSubject() { |  | ||||||
|         return appointmentSubject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentSubject(String appointmentSubject) { |  | ||||||
|         this.appointmentSubject = appointmentSubject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<SectionCell> getAppointmentListSectionCell() { |  | ||||||
|         return listSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateListSectionCell(SectionCell sectionCell) { |  | ||||||
|         listSectionCell.add(sectionCell); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Report getAppointmentReport() { |  | ||||||
|         return report; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentReport(Report report) { |  | ||||||
|         this.report = report; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,123 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
| import jakarta.persistence.Entity; |  | ||||||
| import jakarta.persistence.Table; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "entrepreneur") |  | ||||||
| @PrimaryKeyJoinColumn(name = "idEntrepreneur", referencedColumnName = "idUser") |  | ||||||
| public class Entrepreneur extends User { |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String school; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String course; |  | ||||||
|  |  | ||||||
|     private boolean sneeStatus; |  | ||||||
|  |  | ||||||
|     @ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idProjectParticipation", referencedColumnName = "idProject") |  | ||||||
|     private Project projectParticipation; |  | ||||||
|  |  | ||||||
|     // @Column(insertable=false, updatable=false) |  | ||||||
|     @OneToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idProjectProposed", referencedColumnName = "idProject") |  | ||||||
|     private Project projectProposed; |  | ||||||
|  |  | ||||||
|     /*@ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idAppointment") |  | ||||||
|     private Appointment appointmentEntrepreneur;*/ |  | ||||||
|     // should now be useless |  | ||||||
|  |  | ||||||
|     @OneToOne(mappedBy = "entrepreneurAppointment", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private MakeAppointment makeAppointment; |  | ||||||
|  |  | ||||||
|     public Entrepreneur() {} |  | ||||||
|  |  | ||||||
|     public Entrepreneur( |  | ||||||
|             String userSurname, |  | ||||||
|             String username, |  | ||||||
|             String primaryMail, |  | ||||||
|             String secondaryMail, |  | ||||||
|             String phoneNumber, |  | ||||||
|             String school, |  | ||||||
|             String course, |  | ||||||
|             boolean sneeStatus) { |  | ||||||
|         super(userSurname, username, primaryMail, secondaryMail, phoneNumber); |  | ||||||
|         this.school = school; |  | ||||||
|         this.course = course; |  | ||||||
|         this.sneeStatus = sneeStatus; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Entrepreneur( |  | ||||||
|             Long idUser, |  | ||||||
|             String userSurname, |  | ||||||
|             String userName, |  | ||||||
|             String primaryMail, |  | ||||||
|             String secondaryMail, |  | ||||||
|             String phoneNumber, |  | ||||||
|             String school, |  | ||||||
|             String course, |  | ||||||
|             boolean sneeStatus, |  | ||||||
|             Project projectParticipation, |  | ||||||
|             Project projectProposed, |  | ||||||
|             MakeAppointment makeAppointment) { |  | ||||||
|         super(idUser, userSurname, userName, primaryMail, secondaryMail, phoneNumber); |  | ||||||
|         this.school = school; |  | ||||||
|         this.course = course; |  | ||||||
|         this.sneeStatus = sneeStatus; |  | ||||||
|         this.projectParticipation = projectParticipation; |  | ||||||
|         this.projectProposed = projectProposed; |  | ||||||
|         this.makeAppointment = makeAppointment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getSchool() { |  | ||||||
|         return school; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setSchool(String school) { |  | ||||||
|         this.school = school; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getCourse() { |  | ||||||
|         return course; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setCourse(String course) { |  | ||||||
|         this.course = course; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public boolean isSneeStatus() { |  | ||||||
|         return sneeStatus; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setSneeStatus(boolean statusSnee) { |  | ||||||
|         this.sneeStatus = sneeStatus; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project getProjectParticipation() { |  | ||||||
|         return projectParticipation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setProjectParticipation(Project projectParticipation) { |  | ||||||
|         this.projectParticipation = projectParticipation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project getProjectProposed() { |  | ||||||
|         return projectProposed; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setProjectProposed(Project projectProposed) { |  | ||||||
|         this.projectProposed = projectProposed; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public MakeAppointment getMakeAppointment() { |  | ||||||
|         return makeAppointment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setMakeAppointment(MakeAppointment makeAppointment) { |  | ||||||
|         this.makeAppointment = makeAppointment; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "make_appointment") |  | ||||||
| public class MakeAppointment { |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idMakeAppointment; |  | ||||||
|  |  | ||||||
|     @OneToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idAdministrator") |  | ||||||
|     private Administrator administratorAppointment; |  | ||||||
|  |  | ||||||
|     @OneToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idEntrepreneur") |  | ||||||
|     private Entrepreneur entrepreneurAppointment; |  | ||||||
|  |  | ||||||
|     public MakeAppointment() {} |  | ||||||
|  |  | ||||||
|     public MakeAppointment(Long idMakeAppointment) { |  | ||||||
|         this.idMakeAppointment = idMakeAppointment; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,140 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
|  |  | ||||||
| import java.time.LocalDate; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "project") |  | ||||||
| public class Project { |  | ||||||
|  |  | ||||||
|     @OneToMany(mappedBy = "projectParticipation", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<Entrepreneur> listEntrepreneurParticipation = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     @OneToMany(mappedBy = "projectSectionCell", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<SectionCell> listSectionCell = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idProject; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String projectName; |  | ||||||
|  |  | ||||||
|     private byte[] logo; |  | ||||||
|     private LocalDate creationDate; |  | ||||||
|  |  | ||||||
|     @Column private ProjectDecisionValue projectStatus; |  | ||||||
|  |  | ||||||
|     @ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idAdministrator") |  | ||||||
|     private Administrator projectAdministrator; |  | ||||||
|  |  | ||||||
|     @OneToOne(mappedBy = "projectProposed", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private Entrepreneur entrepreneurProposed; |  | ||||||
|  |  | ||||||
|     public Project() {} |  | ||||||
|  |  | ||||||
|     public Project( |  | ||||||
|             String projectName, |  | ||||||
|             byte[] logo, |  | ||||||
|             LocalDate creationDate, |  | ||||||
|             ProjectDecisionValue projectStatus, |  | ||||||
|             Administrator projectAdministrator) { |  | ||||||
|         this.projectName = projectName; |  | ||||||
|         this.logo = logo; |  | ||||||
|         this.creationDate = creationDate; |  | ||||||
|         // this.projectStatus = (long) projectStatus.ordinal(); |  | ||||||
|         this.projectStatus = projectStatus; |  | ||||||
|         this.projectAdministrator = projectAdministrator; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project( |  | ||||||
|             String projectName, |  | ||||||
|             byte[] logo, |  | ||||||
|             LocalDate creationDate, |  | ||||||
|             ProjectDecisionValue projectStatus, |  | ||||||
|             Administrator projectAdministrator, |  | ||||||
|             Entrepreneur entrepreneurProposed) { |  | ||||||
|         this.projectName = projectName; |  | ||||||
|         this.logo = logo; |  | ||||||
|         this.creationDate = creationDate; |  | ||||||
|         this.projectStatus = projectStatus; |  | ||||||
|         this.projectAdministrator = projectAdministrator; |  | ||||||
|         this.entrepreneurProposed = entrepreneurProposed; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getIdProject() { |  | ||||||
|         return idProject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setIdProject(Long idProject) { |  | ||||||
|         this.idProject = idProject; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getProjectName() { |  | ||||||
|         return projectName; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setProjectName(String projectName) { |  | ||||||
|         this.projectName = projectName; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public byte[] getLogo() { |  | ||||||
|         return logo; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setLogo(byte[] logo) { |  | ||||||
|         this.logo = logo; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public LocalDate getCreationDate() { |  | ||||||
|         return creationDate; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setCreationDate(LocalDate creationDate) { |  | ||||||
|         this.creationDate = creationDate; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public ProjectDecisionValue getProjectStatus() { |  | ||||||
|         return projectStatus; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setProjectStatus(ProjectDecisionValue projectStatus) { |  | ||||||
|         this.projectStatus = projectStatus; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<Entrepreneur> getListEntrepreneurParticipation() { |  | ||||||
|         return listEntrepreneurParticipation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateListEntrepreneurParticipation(Entrepreneur projectParticipant) { |  | ||||||
|         listEntrepreneurParticipation.add(projectParticipant); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<SectionCell> getListSectionCell() { |  | ||||||
|         return listSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateListSectionCell(SectionCell projectSectionCell) { |  | ||||||
|         listSectionCell.add(projectSectionCell); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Administrator getProjectAdministrator() { |  | ||||||
|         return projectAdministrator; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setProjectAdministrator(Administrator projectAdministrator) { |  | ||||||
|         this.projectAdministrator = projectAdministrator; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Entrepreneur getEntrepreneurProposed() { |  | ||||||
|         return entrepreneurProposed; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setEntrepreneurProposed(Entrepreneur entrepreneurProposed) { |  | ||||||
|         this.entrepreneurProposed = entrepreneurProposed; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,25 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| public class ProjectDecision { |  | ||||||
|     public long projectId; |  | ||||||
|     public long adminId; |  | ||||||
|     public long isAccepted; |  | ||||||
|  |  | ||||||
|     public ProjectDecision(long projectId, long adminId, long isAccepted) { |  | ||||||
|         this.projectId = projectId; |  | ||||||
|         this.adminId = adminId; |  | ||||||
|         this.isAccepted = isAccepted; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public String toString() { |  | ||||||
|         return "ProjectDecision{" |  | ||||||
|                 + "projectId=" |  | ||||||
|                 + projectId |  | ||||||
|                 + ", adminId=" |  | ||||||
|                 + adminId |  | ||||||
|                 + ", isAccepted=" |  | ||||||
|                 + isAccepted |  | ||||||
|                 + '}'; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| public enum ProjectDecisionValue { |  | ||||||
|     PENDING, |  | ||||||
|     ACTIVE, |  | ||||||
|     ENDED, |  | ||||||
|     ABORTED, |  | ||||||
|     REJECTED, |  | ||||||
| } |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
| import jakarta.persistence.Entity; |  | ||||||
| import jakarta.persistence.Id; |  | ||||||
| import jakarta.persistence.Table; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "report") |  | ||||||
| public class Report { |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idReport; |  | ||||||
|  |  | ||||||
|     private String reportContent; |  | ||||||
|  |  | ||||||
|     @OneToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idAppointment") |  | ||||||
|     private Appointment appointmentReport; |  | ||||||
|  |  | ||||||
|     public Report() {} |  | ||||||
|  |  | ||||||
|     public Report(Long idReport, String reportContent) { |  | ||||||
|         this.idReport = idReport; |  | ||||||
|         this.reportContent = reportContent; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getIdReport() { |  | ||||||
|         return idReport; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getReportContent() { |  | ||||||
|         return reportContent; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setReportContent(String reportContent) { |  | ||||||
|         this.reportContent = reportContent; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Appointment getAppointmentReport() { |  | ||||||
|         return appointmentReport; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAppointmentReport(Appointment appointmentReport) { |  | ||||||
|         this.appointmentReport = appointmentReport; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| public class RoleRepresentation { |  | ||||||
|     public String id; |  | ||||||
|     public String name; |  | ||||||
|     public String description; |  | ||||||
| } |  | ||||||
| @@ -1,110 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
|  |  | ||||||
| import java.time.LocalDateTime; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "section_cell") |  | ||||||
| public class SectionCell { |  | ||||||
|  |  | ||||||
|     @ManyToMany(mappedBy = "listSectionCell") |  | ||||||
|     private final List<Appointment> listAppointment = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     @OneToMany(mappedBy = "sectionCellAnnotation", fetch = FetchType.LAZY, orphanRemoval = true) |  | ||||||
|     private final List<Annotation> listAnnotation = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idSectionCell; |  | ||||||
|  |  | ||||||
|     @Column() private long sectionId; |  | ||||||
|     private String contentSectionCell; |  | ||||||
|  |  | ||||||
|     /*@ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idAdministrator") |  | ||||||
|     private Administrator administratorSectionCell;*/ |  | ||||||
|     // should now be useless |  | ||||||
|     private LocalDateTime modificationDate; |  | ||||||
|  |  | ||||||
|     @ManyToOne(fetch = FetchType.LAZY) |  | ||||||
|     @JoinColumn(name = "idProject") |  | ||||||
|     private Project projectSectionCell; |  | ||||||
|  |  | ||||||
|     public SectionCell() {} |  | ||||||
|  |  | ||||||
|     public SectionCell( |  | ||||||
|             Long idSectionCell, |  | ||||||
|             Long sectionId, |  | ||||||
|             String contentSectionCell, |  | ||||||
|             LocalDateTime modificationDate, |  | ||||||
|             Project projectSectionCell) { |  | ||||||
|         this.idSectionCell = idSectionCell; |  | ||||||
|         this.sectionId = sectionId; |  | ||||||
|         this.contentSectionCell = contentSectionCell; |  | ||||||
|         this.modificationDate = modificationDate; |  | ||||||
|         this.projectSectionCell = projectSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getIdSectionCell() { |  | ||||||
|         return idSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setIdSectionCell(Long idSectionCell) { |  | ||||||
|         this.idSectionCell = idSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getSectionId() { |  | ||||||
|         return sectionId; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setSectionId(Long sectionId) { |  | ||||||
|         this.sectionId = sectionId; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getContentSectionCell() { |  | ||||||
|         return contentSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setContentSectionCell(String contentSectionCell) { |  | ||||||
|         this.contentSectionCell = contentSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public LocalDateTime getModificationDate() { |  | ||||||
|         return modificationDate; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setModificationDate(LocalDateTime modificationDate) { |  | ||||||
|         this.modificationDate = modificationDate; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project getProjectSectionCell() { |  | ||||||
|         return projectSectionCell; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<Appointment> getAppointmentSectionCell() { |  | ||||||
|         return listAppointment; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateAppointmentSectionCell(Appointment appointment) { |  | ||||||
|         listAppointment.add(appointment); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<Annotation> getListAnnotation() { |  | ||||||
|         return listAnnotation; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void updateListAnnotation(Annotation annotation) { |  | ||||||
|         listAnnotation.add(annotation); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setSectionId(long sectionId) { |  | ||||||
|         this.sectionId = sectionId; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setProjectSectionCell(Project projectSectionCell) { |  | ||||||
|         this.projectSectionCell = projectSectionCell; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,108 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| import jakarta.persistence.*; |  | ||||||
|  |  | ||||||
| @Entity |  | ||||||
| @Table(name = "user_inpulse") |  | ||||||
| @Inheritance(strategy = InheritanceType.JOINED) |  | ||||||
| public class User { |  | ||||||
|  |  | ||||||
|     @Id |  | ||||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) |  | ||||||
|     private Long idUser; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String userSurname; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String userName; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String primaryMail; |  | ||||||
|  |  | ||||||
|     @Column(length = 255) |  | ||||||
|     private String secondaryMail; |  | ||||||
|  |  | ||||||
|     @Column(length = 20) |  | ||||||
|     private String phoneNumber; |  | ||||||
|  |  | ||||||
|     public User() {} |  | ||||||
|  |  | ||||||
|     // TODO: this should be removed as we shouldn't be able to chose the ID. Leaving it for |  | ||||||
|     // compatibility purposes, as soon as it's not used anymore, delete it |  | ||||||
|     public User( |  | ||||||
|             Long idUser, |  | ||||||
|             String userSurname, |  | ||||||
|             String userName, |  | ||||||
|             String primaryMail, |  | ||||||
|             String secondaryMail, |  | ||||||
|             String phoneNumber) { |  | ||||||
|         this.idUser = idUser; |  | ||||||
|         this.userSurname = userSurname; |  | ||||||
|         this.userName = userName; |  | ||||||
|         this.primaryMail = primaryMail; |  | ||||||
|         this.secondaryMail = secondaryMail; |  | ||||||
|         this.phoneNumber = phoneNumber; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public User( |  | ||||||
|             String userSurname, |  | ||||||
|             String userName, |  | ||||||
|             String primaryMail, |  | ||||||
|             String secondaryMail, |  | ||||||
|             String phoneNumber) { |  | ||||||
|         this.userSurname = userSurname; |  | ||||||
|         this.userName = userName; |  | ||||||
|         this.primaryMail = primaryMail; |  | ||||||
|         this.secondaryMail = secondaryMail; |  | ||||||
|         this.phoneNumber = phoneNumber; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getIdUser() { |  | ||||||
|         return idUser; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setIdUser(Long idUser) { |  | ||||||
|         this.idUser = idUser; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getUserSurname() { |  | ||||||
|         return userSurname; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setUserSurname(String userSurname) { |  | ||||||
|         userSurname = userSurname; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getUserName() { |  | ||||||
|         return userName; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setUserName(String userName) { |  | ||||||
|         userName = userName; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getPrimaryMail() { |  | ||||||
|         return primaryMail; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setPrimaryMail(String primaryMail) { |  | ||||||
|         this.primaryMail = primaryMail; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getSecondaryMail() { |  | ||||||
|         return secondaryMail; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setSecondaryMail(String secondaryMail) { |  | ||||||
|         this.secondaryMail = secondaryMail; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getPhoneNumber() { |  | ||||||
|         return phoneNumber; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setPhoneNumber(String phoneNumber) { |  | ||||||
|         phoneNumber = phoneNumber; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| package enseirb.myinpulse.model; |  | ||||||
|  |  | ||||||
| public class UserRepresentation { |  | ||||||
|     public String id; |  | ||||||
|     public String name; |  | ||||||
| } |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Administrator; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface AdministratorRepository extends JpaRepository<Administrator, Long> { |  | ||||||
|  |  | ||||||
|     /* @Query("SELECT a from Administrators a") |  | ||||||
|     Administrator findAllAdministrator(); */ |  | ||||||
|  |  | ||||||
|     Optional<Administrator> findByPrimaryMail(String PrimaryMail); |  | ||||||
| } |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Annotation; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface AnnotationRepository extends JpaRepository<Annotation, Long> {} |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Appointment; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface AppointmentRepository extends JpaRepository<Appointment, Long> {} |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Entrepreneur; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface EntrepreneurRepository extends JpaRepository<Entrepreneur, Long> { |  | ||||||
|  |  | ||||||
|     Iterable<Entrepreneur> getEntrepreneurByProjectParticipation(Project project); |  | ||||||
|  |  | ||||||
|     /* @Query("SELECT e from Entrepreneur e") |  | ||||||
|     Entrepreneur findAllEntrepreneurl(); */ |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.MakeAppointment; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface MakeAppointmentRepository extends JpaRepository<MakeAppointment, Long> {} |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Administrator; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.ProjectDecisionValue; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface ProjectRepository extends JpaRepository<Project, Long> { |  | ||||||
|     Iterable<Project> findByProjectAdministrator(Administrator administrator); |  | ||||||
|  |  | ||||||
|     Iterable<Project> findByProjectStatus(ProjectDecisionValue status); |  | ||||||
|  |  | ||||||
|     Optional<Project> findByProjectName(String projectName); |  | ||||||
| } |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Report; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface ReportRepository extends JpaRepository<Report, Long> {} |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.SectionCell; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| import java.time.LocalDateTime; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface SectionCellRepository extends JpaRepository<SectionCell, Long> { |  | ||||||
|  |  | ||||||
|     Iterable<SectionCell> findByProjectSectionCellAndSectionId(Project project, long sectionId); |  | ||||||
|  |  | ||||||
|     Iterable<SectionCell> findByProjectSectionCellAndSectionIdAndModificationDateBefore( |  | ||||||
|             Project project, long sectionId, LocalDateTime date); |  | ||||||
| } |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| package enseirb.myinpulse.repository; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.User; |  | ||||||
|  |  | ||||||
| import org.springframework.data.jpa.repository.JpaRepository; |  | ||||||
| import org.springframework.data.rest.core.annotation.RepositoryRestResource; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @RepositoryRestResource |  | ||||||
| public interface UserRepository extends JpaRepository<User, Long> { |  | ||||||
|     Optional<User> findByPrimaryMail(String email); |  | ||||||
|  |  | ||||||
|     /* @Query("SELECT u from User u") |  | ||||||
|     User findAllUser(); */ |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -0,0 +1,98 @@ | |||||||
|  | package enseirb.myinpulse.security; | ||||||
|  |  | ||||||
|  | import org.springframework.core.convert.converter.Converter; | ||||||
|  | import org.springframework.security.authentication.AbstractAuthenticationToken; | ||||||
|  | import org.springframework.security.core.GrantedAuthority; | ||||||
|  | import org.springframework.security.core.authority.SimpleGrantedAuthority; | ||||||
|  | import org.springframework.security.oauth2.jwt.Jwt; | ||||||
|  | import org.springframework.security.oauth2.server.resource.authentication.JwtAuthenticationToken; | ||||||
|  | import org.springframework.security.oauth2.server.resource.authentication.JwtGrantedAuthoritiesConverter; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Collection; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.stream.Collectors; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
|  | import static java.util.stream.Collectors.toSet; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | public class KeycloakJwtRolesConverter implements Converter<Jwt, AbstractAuthenticationToken> { | ||||||
|  |     /** | ||||||
|  |      * Prefix used for realm level roles. | ||||||
|  |      */ | ||||||
|  |     public static final String PREFIX_REALM_ROLE = "ROLE_REALM_"; | ||||||
|  |     /** | ||||||
|  |      * Prefix used in combination with the resource (client) name for resource level roles. | ||||||
|  |      */ | ||||||
|  |     public static final String PREFIX_RESOURCE_ROLE = "ROLE_"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Name of the claim containing the realm level roles | ||||||
|  |      */ | ||||||
|  |     private static final String CLAIM_REALM_ACCESS = "realm_access"; | ||||||
|  |     /** | ||||||
|  |      * Name of the claim containing the resources (clients) the user has access to. | ||||||
|  |      */ | ||||||
|  |     private static final String CLAIM_RESOURCE_ACCESS = "resource_access"; | ||||||
|  |     /** | ||||||
|  |      * Name of the claim containing roles. (Applicable to realm and resource level.) | ||||||
|  |      */ | ||||||
|  |     private static final String CLAIM_ROLES = "roles"; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public AbstractAuthenticationToken convert(Jwt source) | ||||||
|  |     { | ||||||
|  |         return new JwtAuthenticationToken(source, Stream.concat(new JwtGrantedAuthoritiesConverter().convert(source) | ||||||
|  |                         .stream(), TEMPORARNAME(source).stream()) | ||||||
|  |                 .collect(toSet())); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Extracts the realm and resource level roles from a JWT token distinguishing between them using prefixes. | ||||||
|  |      */ | ||||||
|  |     public Collection<GrantedAuthority> TEMPORARNAME(Jwt jwt) { | ||||||
|  |         // Collection that will hold the extracted roles | ||||||
|  |         Collection<GrantedAuthority> grantedAuthorities = new ArrayList<>(); | ||||||
|  |  | ||||||
|  |         // Realm roles | ||||||
|  |         // Get the part of the access token that holds the roles assigned on realm level | ||||||
|  |         Map<String, Collection<String>> realmAccess = jwt.getClaim(CLAIM_REALM_ACCESS); | ||||||
|  |  | ||||||
|  |         // Verify that the claim exists and is not empty | ||||||
|  |         if (realmAccess != null && !realmAccess.isEmpty()) { | ||||||
|  |             // From the realm_access claim get the roles | ||||||
|  |             Collection<String> roles = realmAccess.get(CLAIM_ROLES); | ||||||
|  |             // Check if any roles are present | ||||||
|  |             if (roles != null && !roles.isEmpty()) { | ||||||
|  |                 // Iterate of the roles and add them to the granted authorities | ||||||
|  |                 Collection<GrantedAuthority> realmRoles = roles.stream() | ||||||
|  |                         // Prefix all realm roles with "ROLE_realm_" | ||||||
|  |                         .map(role -> new SimpleGrantedAuthority(PREFIX_REALM_ROLE + role)) | ||||||
|  |                         .collect(Collectors.toList()); | ||||||
|  |                 grantedAuthorities.addAll(realmRoles); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Resource (client) roles | ||||||
|  |         // A user might have access to multiple resources all containing their own roles. Therefore, it is a map of | ||||||
|  |         // resource each possibly containing a "roles" property. | ||||||
|  |         Map<String, Map<String, Collection<String>>> resourceAccess = jwt.getClaim(CLAIM_RESOURCE_ACCESS); | ||||||
|  |  | ||||||
|  |         // Check if resources are assigned | ||||||
|  |         if (resourceAccess != null && !resourceAccess.isEmpty()) { | ||||||
|  |             // Iterate of all the resources | ||||||
|  |             resourceAccess.forEach((resource, resourceClaims) -> { | ||||||
|  |                 // Iterate of the "roles" claim inside the resource claims | ||||||
|  |                 resourceClaims.get(CLAIM_ROLES).forEach( | ||||||
|  |                         // Add the role to the granted authority prefixed with ROLE_ and the name of the resource | ||||||
|  |                         role -> grantedAuthorities.add(new SimpleGrantedAuthority(PREFIX_RESOURCE_ROLE + resource + "_" + role)) | ||||||
|  |                 ); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return grantedAuthorities; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,166 +0,0 @@ | |||||||
| package enseirb.myinpulse.service; |  | ||||||
|  |  | ||||||
| import static enseirb.myinpulse.model.ProjectDecisionValue.ACTIVE; |  | ||||||
| import static enseirb.myinpulse.model.ProjectDecisionValue.REJECTED; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.*; |  | ||||||
| import enseirb.myinpulse.service.database.*; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class AdminApiService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final ProjectService projectService; |  | ||||||
|     private final UserService userService; |  | ||||||
|     private final AdministratorService administratorService; |  | ||||||
|     private final UtilsService utilsService; |  | ||||||
|     private final AppointmentService appointmentService; |  | ||||||
|     private final ReportService reportService; |  | ||||||
|     private final SectionCellService sectionCellService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     AdminApiService( |  | ||||||
|             ProjectService projectService, |  | ||||||
|             UserService userService, |  | ||||||
|             AdministratorService administratorService, |  | ||||||
|             UtilsService utilsService, |  | ||||||
|             AppointmentService appointmentService, |  | ||||||
|             ReportService reportService, |  | ||||||
|             SectionCellService sectionCellService) { |  | ||||||
|         this.projectService = projectService; |  | ||||||
|         this.userService = userService; |  | ||||||
|         this.administratorService = administratorService; |  | ||||||
|         this.utilsService = utilsService; |  | ||||||
|         this.appointmentService = appointmentService; |  | ||||||
|         this.reportService = reportService; |  | ||||||
|         this.sectionCellService = sectionCellService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: check if tests are sufficient - peer verification required |  | ||||||
|     public Iterable<Project> getProjectsOfAdmin(String mail) { |  | ||||||
|         return projectService.getProjectsByAdminId( |  | ||||||
|                 administratorService.getAdministratorById( |  | ||||||
|                         this.userService.getUserByEmail(mail).getIdUser())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Appointment> getUpcomingAppointments(String mail) { |  | ||||||
|         logger.info("User {} check their upcoming appointments", mail); |  | ||||||
|         User user = this.userService.getUserByEmail(mail); |  | ||||||
|         List<Appointment> appointments = new ArrayList<>(); |  | ||||||
|         if (user instanceof Administrator) { |  | ||||||
|             List<Project> projects = new ArrayList<>(((Administrator) user).getListProject()); |  | ||||||
|             projects.forEach( |  | ||||||
|                     project -> { |  | ||||||
|                         project.getListSectionCell() |  | ||||||
|                                 .forEach( |  | ||||||
|                                         sectionCell -> { |  | ||||||
|                                             appointments.addAll( |  | ||||||
|                                                     this.sectionCellService |  | ||||||
|                                                             .getAppointmentsBySectionCellId( |  | ||||||
|                                                                     sectionCell |  | ||||||
|                                                                             .getIdSectionCell())); |  | ||||||
|                                         }); |  | ||||||
|                     }); |  | ||||||
|         } |  | ||||||
|         if (user instanceof Entrepreneur) { |  | ||||||
|             Project project = ((Entrepreneur) user).getProjectParticipation(); |  | ||||||
|             project.getListSectionCell() |  | ||||||
|                     .forEach( |  | ||||||
|                             sectionCell -> { |  | ||||||
|                                 appointments.addAll( |  | ||||||
|                                         this.sectionCellService.getAppointmentsBySectionCellId( |  | ||||||
|                                                 sectionCell.getIdSectionCell())); |  | ||||||
|                             }); |  | ||||||
|         } |  | ||||||
|         return appointments; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: check if tests are sufficient - peer verification required |  | ||||||
|     public Iterable<Project> getPendingProjects() { |  | ||||||
|         return this.projectService.getPendingProjects(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: check if tests are sufficient - peer verification required |  | ||||||
|     public void validateProject(ProjectDecision decision) { |  | ||||||
|         projectService.updateProject( |  | ||||||
|                 decision.projectId, |  | ||||||
|                 null, |  | ||||||
|                 null, |  | ||||||
|                 null, |  | ||||||
|                 (decision.isAccepted == 1) ? ACTIVE : REJECTED, |  | ||||||
|                 this.administratorService.getAdministratorById(decision.adminId)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: check if tests are sufficient - peer verification required |  | ||||||
|     public void addNewProject(Project project) { |  | ||||||
|         project.setIdProject(null); |  | ||||||
|         // We remove the ID from the request to be sure that it will be auto generated |  | ||||||
|         try { |  | ||||||
|             this.projectService.getProjectByName(project.getProjectName(), true); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.CONFLICT, "Project already exists"); |  | ||||||
|         } catch (ResponseStatusException e) { |  | ||||||
|             if (e.getStatusCode() == HttpStatus.CONFLICT) { |  | ||||||
|                 throw new ResponseStatusException(HttpStatus.CONFLICT, "Project already exists"); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         Project newProject = projectService.addNewProject(project); |  | ||||||
|         if (project.getProjectAdministrator() != null) { |  | ||||||
|             newProject.getProjectAdministrator().updateListProject(newProject); |  | ||||||
|         } |  | ||||||
|         if (newProject.getEntrepreneurProposed() != null) { |  | ||||||
|             Entrepreneur proposed = newProject.getEntrepreneurProposed(); |  | ||||||
|             proposed.setProjectProposed(newProject); |  | ||||||
|             proposed.setProjectParticipation(newProject); |  | ||||||
|         } |  | ||||||
|         newProject |  | ||||||
|                 .getListEntrepreneurParticipation() |  | ||||||
|                 .forEach( |  | ||||||
|                         participation -> { |  | ||||||
|                             participation.setProjectParticipation(newProject); |  | ||||||
|                         }); |  | ||||||
|         newProject |  | ||||||
|                 .getListSectionCell() |  | ||||||
|                 .forEach( |  | ||||||
|                         sectionCell -> { |  | ||||||
|                             sectionCell.setProjectSectionCell(newProject); |  | ||||||
|                         }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void createAppointmentReport(long appointmentId, Report report, String mail) { |  | ||||||
|         long projectId = |  | ||||||
|                 this.appointmentService |  | ||||||
|                         .getAppointmentById(appointmentId) |  | ||||||
|                         .getAppointmentListSectionCell() |  | ||||||
|                         .getFirst() |  | ||||||
|                         .getProjectSectionCell() |  | ||||||
|                         .getIdProject(); |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to add an report for appointment {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     projectId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info("User {} added a report for appointment {}", mail, projectId); |  | ||||||
|         Report addedReport = this.reportService.addNewReport(report); |  | ||||||
|         addedReport.setAppointmentReport(this.appointmentService.getAppointmentById(appointmentId)); |  | ||||||
|         this.appointmentService.getAppointmentById(appointmentId).setAppointmentReport(addedReport); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: test |  | ||||||
|     public void deleteProject(long projectId) { |  | ||||||
|         this.projectService.deleteProjectById(projectId); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,135 +0,0 @@ | |||||||
| package enseirb.myinpulse.service; |  | ||||||
|  |  | ||||||
| import static enseirb.myinpulse.model.ProjectDecisionValue.PENDING; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.SectionCell; |  | ||||||
| import enseirb.myinpulse.service.database.ProjectService; |  | ||||||
| import enseirb.myinpulse.service.database.SectionCellService; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class EntrepreneurApiService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final SectionCellService sectionCellService; |  | ||||||
|     private final ProjectService projectService; |  | ||||||
|     private final UtilsService utilsService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     EntrepreneurApiService( |  | ||||||
|             SectionCellService sectionCellService, |  | ||||||
|             ProjectService projectService, |  | ||||||
|             UtilsService utilsService) { |  | ||||||
|         this.sectionCellService = sectionCellService; |  | ||||||
|         this.projectService = projectService; |  | ||||||
|         this.utilsService = utilsService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void editSectionCell(Long sectionCellId, SectionCell sectionCell, String mail) { |  | ||||||
|         SectionCell editSectionCell = sectionCellService.getSectionCellById(sectionCellId); |  | ||||||
|         if (editSectionCell == null) { |  | ||||||
|             System.err.println("Trying to edit unknown section cell"); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (!utilsService.isAllowedToCheckProject( |  | ||||||
|                 mail, this.sectionCellService.getProjectId(sectionCellId))) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to edit section cells {} of the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     sectionCellId, |  | ||||||
|                     this.sectionCellService.getProjectId(sectionCellId)); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info( |  | ||||||
|                 "User {} edited section cell {} of the project with id {}", |  | ||||||
|                 mail, |  | ||||||
|                 sectionCellId, |  | ||||||
|                 this.sectionCellService.getProjectId(sectionCellId)); |  | ||||||
|         sectionCellService.updateSectionCell( |  | ||||||
|                 sectionCellId, |  | ||||||
|                 sectionCell.getSectionId(), |  | ||||||
|                 sectionCell.getContentSectionCell(), |  | ||||||
|                 sectionCell.getModificationDate()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void removeSectionCell(Long sectionCellId, String mail) { |  | ||||||
|         SectionCell editSectionCell = sectionCellService.getSectionCellById(sectionCellId); |  | ||||||
|         if (editSectionCell == null) { |  | ||||||
|             System.err.println("Trying to remove unknown section cell"); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (!utilsService.isAllowedToCheckProject( |  | ||||||
|                 mail, this.sectionCellService.getProjectId(sectionCellId))) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to remove section cells {} of the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     sectionCellId, |  | ||||||
|                     this.sectionCellService.getSectionCellById(sectionCellId)); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info( |  | ||||||
|                 "User {} removed section cell {} of the project with id {}", |  | ||||||
|                 mail, |  | ||||||
|                 sectionCellId, |  | ||||||
|                 this.sectionCellService.getProjectId(sectionCellId)); |  | ||||||
|         sectionCellService.removeSectionCellById(sectionCellId); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void addSectionCell(SectionCell sectionCell, String mail) { |  | ||||||
|         if (sectionCell == null) { |  | ||||||
|             System.err.println("Trying to create an empty section cell"); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.BAD_REQUEST, "La cellule de section fournie est vide"); |  | ||||||
|         } |  | ||||||
|         if (!utilsService.isAllowedToCheckProject( |  | ||||||
|                 mail, this.sectionCellService.getProjectId(sectionCell.getIdSectionCell()))) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to add a section cell to the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     this.sectionCellService.getProjectId(sectionCell.getIdSectionCell())); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info( |  | ||||||
|                 "User {} added a new section cell {} to the project with id {}", |  | ||||||
|                 mail, |  | ||||||
|                 sectionCell.getIdSectionCell(), |  | ||||||
|                 this.sectionCellService.getProjectId(sectionCell.getIdSectionCell())); |  | ||||||
|         SectionCell newSectionCell = sectionCellService.addNewSectionCell(sectionCell); |  | ||||||
|         newSectionCell.getProjectSectionCell().updateListSectionCell(newSectionCell); |  | ||||||
|         newSectionCell |  | ||||||
|                 .getAppointmentSectionCell() |  | ||||||
|                 .forEach( |  | ||||||
|                         appointment -> { |  | ||||||
|                             appointment.updateListSectionCell(newSectionCell); |  | ||||||
|                         }); |  | ||||||
|         newSectionCell |  | ||||||
|                 .getListAnnotation() |  | ||||||
|                 .forEach( |  | ||||||
|                         annotation -> { |  | ||||||
|                             annotation.setSectionCellAnnotation(newSectionCell); |  | ||||||
|                         }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void requestNewProject(Project project, String mail) { |  | ||||||
|         if (project == null) { |  | ||||||
|             logger.error("Trying to request the creation of a null project"); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.BAD_REQUEST, "Le projet fourni est vide"); |  | ||||||
|         } |  | ||||||
|         logger.info("User {} created a new project with id {}", mail, project.getIdProject()); |  | ||||||
|         project.setProjectStatus(PENDING); |  | ||||||
|         projectService.addNewProject(project); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,135 +0,0 @@ | |||||||
| package enseirb.myinpulse.service; |  | ||||||
|  |  | ||||||
| import static org.springframework.http.MediaType.APPLICATION_JSON; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.exception.UserNotFoundException; |  | ||||||
| import enseirb.myinpulse.model.RoleRepresentation; |  | ||||||
| import enseirb.myinpulse.model.UserRepresentation; |  | ||||||
|  |  | ||||||
| import org.springframework.web.client.RestClient; |  | ||||||
|  |  | ||||||
| import javax.management.relation.RoleNotFoundException; |  | ||||||
|  |  | ||||||
| public class KeycloakApi { |  | ||||||
|  |  | ||||||
|     static final String keycloakUrl; |  | ||||||
|     static final String realmName; |  | ||||||
|  |  | ||||||
|     static { |  | ||||||
|         if (System.getenv("VITE_KEYCLOAK_URL") == null) { |  | ||||||
|             System.exit(-1); |  | ||||||
|         } |  | ||||||
|         keycloakUrl = System.getenv("VITE_KEYCLOAK_URL"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static { |  | ||||||
|         if (System.getenv("VITE_KEYCLOAK_REALM") == null) { |  | ||||||
|             System.exit(-1); |  | ||||||
|         } |  | ||||||
|         realmName = System.getenv("VITE_KEYCLOAK_REALM"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Uses Keycloak API to retrieve a role representation of a role by its name |  | ||||||
|      * |  | ||||||
|      * @param roleName name of the role |  | ||||||
|      * @param bearer authorization header used by the client to authenticate to keycloak |  | ||||||
|      */ |  | ||||||
|     public static RoleRepresentation getRoleRepresentationByName(String roleName, String bearer) |  | ||||||
|             throws RoleNotFoundException { |  | ||||||
|         RoleRepresentation[] response = |  | ||||||
|                 RestClient.builder() |  | ||||||
|                         .baseUrl(keycloakUrl) |  | ||||||
|                         .defaultHeader("Authorization", bearer) |  | ||||||
|                         .build() |  | ||||||
|                         .get() |  | ||||||
|                         .uri("/admin/realms/{realmName}/roles/{roleName}", realmName, roleName) |  | ||||||
|                         .retrieve() |  | ||||||
|                         .body(RoleRepresentation[].class); |  | ||||||
|  |  | ||||||
|         if (response == null || response.length == 0) { |  | ||||||
|             throw new RoleNotFoundException("Role not found"); |  | ||||||
|         } |  | ||||||
|         return response[0]; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Use keycloak API to to retreive a userID via his name or email. |  | ||||||
|      * |  | ||||||
|      * @param username username or mail of the user |  | ||||||
|      * @param bearer bearer of the user, allowing access to database |  | ||||||
|      * @return the userid, as a String |  | ||||||
|      * @throws UserNotFoundException |  | ||||||
|      */ |  | ||||||
|     public static String getUserIdByName(String username, String bearer) |  | ||||||
|             throws UserNotFoundException { |  | ||||||
|         UserRepresentation[] response = |  | ||||||
|                 RestClient.builder() |  | ||||||
|                         .baseUrl(keycloakUrl) |  | ||||||
|                         .defaultHeader("Authorization", bearer) |  | ||||||
|                         .build() |  | ||||||
|                         .get() |  | ||||||
|                         .uri( |  | ||||||
|                                 "/admin/realms/{realmName}/users?username={username}", |  | ||||||
|                                 realmName, |  | ||||||
|                                 username) |  | ||||||
|                         .retrieve() |  | ||||||
|                         .body(UserRepresentation[].class); |  | ||||||
|  |  | ||||||
|         if (response == null || response.length == 0) { |  | ||||||
|             throw new UserNotFoundException("User not found"); |  | ||||||
|         } |  | ||||||
|         return response[0].id; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * TODO: check for error |  | ||||||
|      * |  | ||||||
|      * <p>Set a keycloak role to a keycloak user. |  | ||||||
|      * |  | ||||||
|      * <p>Usual roles should be `MyINPulse-admin` and `MyINPulse-entrepreneur` |  | ||||||
|      * |  | ||||||
|      * @param username |  | ||||||
|      * @param roleName |  | ||||||
|      * @param bearer |  | ||||||
|      * @throws RoleNotFoundException |  | ||||||
|      * @throws UserNotFoundException |  | ||||||
|      */ |  | ||||||
|     public static void setRoleToUser(String username, String roleName, String bearer) |  | ||||||
|             throws RoleNotFoundException, UserNotFoundException { |  | ||||||
|         RoleRepresentation roleRepresentation = getRoleRepresentationByName(roleName, bearer); |  | ||||||
|         String userId = getUserIdByName(username, bearer); |  | ||||||
|  |  | ||||||
|         RestClient.builder() |  | ||||||
|                 .baseUrl(keycloakUrl) |  | ||||||
|                 .defaultHeader("Authorization", bearer) |  | ||||||
|                 .build() |  | ||||||
|                 .post() |  | ||||||
|                 .uri( |  | ||||||
|                         "/admin/realms/${realmName}/users/${userId}/role-mappings/realm", |  | ||||||
|                         realmName, |  | ||||||
|                         userId) |  | ||||||
|                 .body(roleRepresentation) |  | ||||||
|                 .contentType(APPLICATION_JSON) |  | ||||||
|                 .retrieve(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Delete a user from Keycloak database. TODO: check the bearer permission. |  | ||||||
|      * |  | ||||||
|      * @param username |  | ||||||
|      * @param bearer |  | ||||||
|      * @throws UserNotFoundException |  | ||||||
|      */ |  | ||||||
|     public static void deleteUser(String username, String bearer) throws UserNotFoundException { |  | ||||||
|         String userId = getUserIdByName(username, bearer); |  | ||||||
|  |  | ||||||
|         RestClient.builder() |  | ||||||
|                 .baseUrl(keycloakUrl) |  | ||||||
|                 .defaultHeader("Authorization", bearer) |  | ||||||
|                 .build() |  | ||||||
|                 .delete() |  | ||||||
|                 .uri("/admin/realms/${realmName}/users/${userId}", realmName, userId) |  | ||||||
|                 .retrieve(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,252 +0,0 @@ | |||||||
| package enseirb.myinpulse.service; |  | ||||||
|  |  | ||||||
| import com.itextpdf.text.*; |  | ||||||
| import com.itextpdf.text.pdf.PdfWriter; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.*; |  | ||||||
| import enseirb.myinpulse.service.database.*; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.FileOutputStream; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.net.URI; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
| import java.nio.file.Files; |  | ||||||
| import java.nio.file.Paths; |  | ||||||
| import java.nio.file.StandardCopyOption; |  | ||||||
| import java.time.LocalDateTime; |  | ||||||
| import java.time.format.DateTimeFormatter; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class SharedApiService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final ProjectService projectService; |  | ||||||
|     private final EntrepreneurService entrepreneurService; |  | ||||||
|     private final SectionCellService sectionCellService; |  | ||||||
|     private final AppointmentService appointmentService; |  | ||||||
|  |  | ||||||
|     private final UtilsService utilsService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     SharedApiService( |  | ||||||
|             ProjectService projectService, |  | ||||||
|             EntrepreneurService entrepreneurService, |  | ||||||
|             SectionCellService sectionCellService, |  | ||||||
|             AppointmentService appointmentService, |  | ||||||
|             UtilsService utilsService) { |  | ||||||
|         this.projectService = projectService; |  | ||||||
|         this.entrepreneurService = entrepreneurService; |  | ||||||
|         this.sectionCellService = sectionCellService; |  | ||||||
|         this.appointmentService = appointmentService; |  | ||||||
|         this.utilsService = utilsService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO filter this with date |  | ||||||
|     public Iterable<SectionCell> getSectionCells( |  | ||||||
|             long projectId, long sectionId, String date, String mail) { |  | ||||||
|  |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to check section cells of the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     projectId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm"); |  | ||||||
|         LocalDateTime dateTime = LocalDateTime.parse(date, formatter); |  | ||||||
|  |  | ||||||
|         Project project = this.projectService.getProjectById(projectId); |  | ||||||
|         return this.sectionCellService.getSectionCellsByProjectAndSectionIdBeforeDate( |  | ||||||
|                 project, sectionId, dateTime); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: test |  | ||||||
|     public Iterable<Entrepreneur> getEntrepreneursByProjectId(long projectId, String mail) { |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to check the member of the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     projectId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         Project project = this.projectService.getProjectById(projectId); |  | ||||||
|         return this.entrepreneurService.GetEntrepreneurByProject(project); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: test |  | ||||||
|     public Administrator getAdminByProjectId(long projectId, String mail) { |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to check the admin of the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     projectId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         Project project = this.projectService.getProjectById(projectId); |  | ||||||
|         return project.getProjectAdministrator(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Appointment> getAppointmentsByProjectId(long projectId, String mail) { |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to check the appointments related to the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     projectId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info( |  | ||||||
|                 "User {} tried to check the appointments related to the project {}", |  | ||||||
|                 mail, |  | ||||||
|                 projectId); |  | ||||||
|         Iterable<SectionCell> sectionCells = |  | ||||||
|                 this.sectionCellService.getSectionCellsByProject( |  | ||||||
|                         projectService.getProjectById(projectId), |  | ||||||
|                         2L); // sectionId useless in this function ? |  | ||||||
|         List<Appointment> appointments = new ArrayList<Appointment>(); |  | ||||||
|         sectionCells.forEach( |  | ||||||
|                 sectionCell -> { |  | ||||||
|                     appointments.addAll( |  | ||||||
|                             this.sectionCellService.getAppointmentsBySectionCellId( |  | ||||||
|                                     sectionCell.getIdSectionCell())); |  | ||||||
|                 }); |  | ||||||
|         return appointments; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void getPDFReport(long appointmentId, String mail) |  | ||||||
|             throws DocumentException, URISyntaxException, IOException { |  | ||||||
|         long projectId = |  | ||||||
|                 this.appointmentService |  | ||||||
|                         .getAppointmentById(appointmentId) |  | ||||||
|                         .getAppointmentListSectionCell() |  | ||||||
|                         .getFirst() |  | ||||||
|                         .getProjectSectionCell() |  | ||||||
|                         .getIdProject(); |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to generate the PDF report {} related to the appointment {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     this.appointmentService |  | ||||||
|                             .getAppointmentById(appointmentId) |  | ||||||
|                             .getAppointmentReport() |  | ||||||
|                             .getIdReport(), |  | ||||||
|                     appointmentId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info( |  | ||||||
|                 "User {} generated the PDF report related to appointment {}", mail, appointmentId); |  | ||||||
|  |  | ||||||
|         String reportContent = |  | ||||||
|                 this.appointmentService |  | ||||||
|                         .getAppointmentById(appointmentId) |  | ||||||
|                         .getAppointmentReport() |  | ||||||
|                         .getReportContent(); |  | ||||||
|  |  | ||||||
|         // PDF generation |  | ||||||
|         Document document = new Document(); |  | ||||||
|         PdfWriter.getInstance(document, new FileOutputStream("Report" + appointmentId + ".pdf")); |  | ||||||
|         document.open(); |  | ||||||
|  |  | ||||||
|         Paragraph title = |  | ||||||
|                 new Paragraph( |  | ||||||
|                         new Phrase( |  | ||||||
|                                 "Compte Rendu - Réunion du " |  | ||||||
|                                         + this.appointmentService |  | ||||||
|                                                 .getAppointmentById(appointmentId) |  | ||||||
|                                                 .getAppointmentDate() |  | ||||||
|                                                 .toString(), |  | ||||||
|                                 FontFactory.getFont( |  | ||||||
|                                         FontFactory.HELVETICA, |  | ||||||
|                                         20, |  | ||||||
|                                         Font.BOLDITALIC, |  | ||||||
|                                         BaseColor.BLACK))); |  | ||||||
|         title.setAlignment(Element.ALIGN_CENTER); |  | ||||||
|         document.add(title); |  | ||||||
|  |  | ||||||
|         Font subsection = |  | ||||||
|                 FontFactory.getFont(FontFactory.HELVETICA, 14, Font.UNDERLINE, BaseColor.DARK_GRAY); |  | ||||||
|         Font body = FontFactory.getFont(FontFactory.COURIER, 12, BaseColor.BLACK); |  | ||||||
|  |  | ||||||
|         String[] split = reportContent.split(" "); |  | ||||||
|  |  | ||||||
|         String tmp = ""; |  | ||||||
|         int counter = 1; |  | ||||||
|         for (String s : split) { |  | ||||||
|             if (s.equals("//")) { |  | ||||||
|                 Chunk chunk = new Chunk(tmp, body); |  | ||||||
|                 document.add(chunk); |  | ||||||
|                 document.add(new Paragraph("\n")); |  | ||||||
|                 tmp = ""; |  | ||||||
|                 Paragraph paragraph = new Paragraph("Point n°" + counter + " : ", subsection); |  | ||||||
|                 document.add(paragraph); |  | ||||||
|                 document.add(new Paragraph("\n")); |  | ||||||
|                 counter++; |  | ||||||
|             } else { |  | ||||||
|                 tmp = tmp.concat(s + " "); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         Chunk chunk = new Chunk(tmp, body); |  | ||||||
|         document.add(chunk); |  | ||||||
|         document.add(new Paragraph("\n")); |  | ||||||
|  |  | ||||||
|         document.close(); |  | ||||||
|  |  | ||||||
|         // Replace uri with website address |  | ||||||
|         Files.copy( |  | ||||||
|                 new URI( |  | ||||||
|                                 "http://localhost:8080/shared/projects/appointments/report/" |  | ||||||
|                                         + appointmentId) |  | ||||||
|                         .toURL() |  | ||||||
|                         .openStream(), |  | ||||||
|                 Paths.get("Report" + appointmentId + ".pdf"), |  | ||||||
|                 StandardCopyOption.REPLACE_EXISTING); |  | ||||||
|  |  | ||||||
|         // delete file, we don't want to stock all reports on the server |  | ||||||
|         File file = new File("Report" + appointmentId + ".pdf"); |  | ||||||
|         if (!file.delete()) { |  | ||||||
|             logger.warn("Failed to delete report {}", file.getAbsolutePath()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void createAppointmentRequest(Appointment appointment, String mail) { |  | ||||||
|         long projectId = |  | ||||||
|                 appointment |  | ||||||
|                         .getAppointmentListSectionCell() |  | ||||||
|                         .getFirst() |  | ||||||
|                         .getProjectSectionCell() |  | ||||||
|                         .getIdProject(); |  | ||||||
|         if (!utilsService.isAllowedToCheckProject(mail, projectId)) { |  | ||||||
|             logger.warn( |  | ||||||
|                     "User {} tried to create for the project {} but is not allowed to.", |  | ||||||
|                     mail, |  | ||||||
|                     projectId); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.UNAUTHORIZED, "You're not allowed to check this project"); |  | ||||||
|         } |  | ||||||
|         logger.info("User {} tried to create an appointment for project {}", mail, projectId); |  | ||||||
|         Appointment newAppointment = this.appointmentService.addNewAppointment(appointment); |  | ||||||
|         newAppointment |  | ||||||
|                 .getAppointmentListSectionCell() |  | ||||||
|                 .forEach( |  | ||||||
|                         sectionCell -> { |  | ||||||
|                             sectionCell.updateAppointmentSectionCell(newAppointment); |  | ||||||
|                         }); |  | ||||||
|         newAppointment.getAppointmentReport().setAppointmentReport(newAppointment); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,62 +0,0 @@ | |||||||
| package enseirb.myinpulse.service; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Administrator; |  | ||||||
| import enseirb.myinpulse.model.Entrepreneur; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.User; |  | ||||||
| import enseirb.myinpulse.service.database.AdministratorService; |  | ||||||
| import enseirb.myinpulse.service.database.EntrepreneurService; |  | ||||||
| import enseirb.myinpulse.service.database.ProjectService; |  | ||||||
| import enseirb.myinpulse.service.database.UserService; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class UtilsService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final UserService userService; |  | ||||||
|     private final ProjectService projectService; |  | ||||||
|     private final EntrepreneurService entrepreneurService; |  | ||||||
|     private final AdministratorService administratorService; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     UtilsService( |  | ||||||
|             ProjectService projectService, |  | ||||||
|             UserService userService, |  | ||||||
|             EntrepreneurService entrepreneurService, |  | ||||||
|             AdministratorService administratorService) { |  | ||||||
|         this.userService = userService; |  | ||||||
|         this.projectService = projectService; |  | ||||||
|         this.entrepreneurService = entrepreneurService; |  | ||||||
|         this.administratorService = administratorService; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: test? |  | ||||||
|     public Boolean isAllowedToCheckProject(String mail, long projectId) { |  | ||||||
|         if (isAnAdmin(mail)) { |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         User user = this.userService.getUserByEmail(mail); |  | ||||||
|         Entrepreneur entrepreneur = this.entrepreneurService.getEntrepreneurById(user.getIdUser()); |  | ||||||
|         Project project = this.projectService.getProjectById(projectId); |  | ||||||
|         return entrepreneur.getProjectParticipation() == project; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: test |  | ||||||
|     Boolean isAnAdmin(String mail) { |  | ||||||
|         try { |  | ||||||
|             long userId = this.userService.getUserByEmail(mail).getIdUser(); |  | ||||||
|             Administrator a = this.administratorService.getAdministratorById(userId); |  | ||||||
|             return true; |  | ||||||
|         } catch (ResponseStatusException e) { |  | ||||||
|             logger.info(e); |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,60 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Administrator; |  | ||||||
| import enseirb.myinpulse.repository.AdministratorRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class AdministratorService { |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final AdministratorRepository administratorRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     AdministratorService(AdministratorRepository administratorRepository) { |  | ||||||
|         this.administratorRepository = administratorRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Administrator> allAdministrators() { |  | ||||||
|         return this.administratorRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Administrator getAdministratorById(long id) { |  | ||||||
|         Optional<Administrator> administrator = this.administratorRepository.findById(id); |  | ||||||
|         if (administrator.isEmpty()) { |  | ||||||
|             logger.error("No administrator found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cet administrateur n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return administrator.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Administrator getAdministratorByPrimaryMain(String primaryMail) { |  | ||||||
|         Optional<Administrator> administrator = |  | ||||||
|                 this.administratorRepository.findByPrimaryMail(primaryMail); |  | ||||||
|         if (administrator.isEmpty()) { |  | ||||||
|             logger.error("No administrator found with the mail {}", primaryMail); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cet administrateur n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return administrator.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Administrator addAdministrator(Administrator administrator) { |  | ||||||
|         return this.administratorRepository.save(administrator); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /* |  | ||||||
|     public Administrator getAdministratorByProject(Project project) { |  | ||||||
|         r |  | ||||||
|     } |  | ||||||
|      */ |  | ||||||
| } |  | ||||||
| @@ -1,61 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Annotation; |  | ||||||
| import enseirb.myinpulse.repository.AnnotationRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class AnnotationService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final AnnotationRepository annotationRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     AnnotationService(AnnotationRepository annotationRepository) { |  | ||||||
|         this.annotationRepository = annotationRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Annotation> getAllAnnotations() { |  | ||||||
|         return annotationRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Annotation getAnnotationById(Long id) { |  | ||||||
|         Optional<Annotation> annotation = annotationRepository.findById(id); |  | ||||||
|         if (annotation.isEmpty()) { |  | ||||||
|             logger.error("getAnnotationById : No annotation found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cette annotation n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return annotation.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Annotation addNewAnnotation(Annotation annotation) { |  | ||||||
|         return this.annotationRepository.save(annotation); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void deleteAnnotationById(Long id) { |  | ||||||
|         this.annotationRepository.deleteById(id); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Annotation updateAnnotation(Long id, String comment) { |  | ||||||
|         Optional<Annotation> annotation = annotationRepository.findById(id); |  | ||||||
|         if (annotation.isEmpty()) { |  | ||||||
|             logger.error("updateAnnotation : No annotation found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cette annotation n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (comment != null) { |  | ||||||
|             annotation.get().setComment(comment); |  | ||||||
|         } |  | ||||||
|         return this.annotationRepository.save(annotation.get()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,79 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Appointment; |  | ||||||
| import enseirb.myinpulse.repository.AppointmentRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.time.LocalDate; |  | ||||||
| import java.time.LocalTime; |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class AppointmentService { |  | ||||||
|  |  | ||||||
|     private static final Logger logger = LogManager.getLogger(AppointmentService.class); |  | ||||||
|  |  | ||||||
|     private AppointmentRepository appointmentRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     AppointmentService(AppointmentRepository appointmentRepository) { |  | ||||||
|         this.appointmentRepository = appointmentRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Appointment> getallAppointments() { |  | ||||||
|         return this.appointmentRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Appointment getAppointmentById(Long id) { |  | ||||||
|         Optional<Appointment> appointment = this.appointmentRepository.findById(id); |  | ||||||
|         if (appointment.isEmpty()) { |  | ||||||
|             logger.error("getAppointmentById : No appointment found with id {}", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce rendez vous n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return appointment.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Appointment addNewAppointment(Appointment appointment) { |  | ||||||
|         return this.appointmentRepository.save(appointment); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void deleteAppointmentById(Long id) { |  | ||||||
|         this.appointmentRepository.deleteById(id); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Appointment updateAppointment( |  | ||||||
|             Long id, |  | ||||||
|             LocalDate appointmentDate, |  | ||||||
|             LocalTime appointmentTime, |  | ||||||
|             LocalTime appointmentDuration, |  | ||||||
|             String appointmentPlace, |  | ||||||
|             String appointmentSubject) { |  | ||||||
|         Optional<Appointment> appointment = this.appointmentRepository.findById(id); |  | ||||||
|         if (appointment.isEmpty()) { |  | ||||||
|             logger.error("updateAppointment : No appointment found with id {}", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce rendez vous n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (appointmentDate != null) { |  | ||||||
|             appointment.get().setAppointmentDate(appointmentDate); |  | ||||||
|         } |  | ||||||
|         if (appointmentTime != null) { |  | ||||||
|             appointment.get().setAppointmentTime(appointmentTime); |  | ||||||
|         } |  | ||||||
|         if (appointmentDuration != null) { |  | ||||||
|             appointment.get().setAppointmentDuration(appointmentDuration); |  | ||||||
|         } |  | ||||||
|         if (appointmentPlace != null) { |  | ||||||
|             appointment.get().setAppointmentPlace(appointmentPlace); |  | ||||||
|         } |  | ||||||
|         if (appointmentSubject != null) { |  | ||||||
|             appointment.get().setAppointmentSubject(appointmentSubject); |  | ||||||
|         } |  | ||||||
|         return this.appointmentRepository.save(appointment.get()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,67 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Entrepreneur; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.repository.EntrepreneurRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class EntrepreneurService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final EntrepreneurRepository entrepreneurRepository; |  | ||||||
|  |  | ||||||
|     EntrepreneurService(EntrepreneurRepository entrepreneurRepository) { |  | ||||||
|         this.entrepreneurRepository = entrepreneurRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Entrepreneur> getAllEntrepreneurs() { |  | ||||||
|         return this.entrepreneurRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Entrepreneur getEntrepreneurById(Long id) { |  | ||||||
|         Optional<Entrepreneur> entrepreneur = entrepreneurRepository.findById(id); |  | ||||||
|         if (entrepreneur.isEmpty()) { |  | ||||||
|             logger.error("getEntrepreneurById : No entrepreneur found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cet entrepreneur n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return entrepreneur.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Entrepreneur addEntrepreneur(Entrepreneur entrepreneur) { |  | ||||||
|         return this.entrepreneurRepository.save(entrepreneur); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Entrepreneur updateEntrepreneur( |  | ||||||
|             Long id, String school, String course, Boolean sneeStatus) { |  | ||||||
|         Optional<Entrepreneur> entrepreneur = entrepreneurRepository.findById(id); |  | ||||||
|         if (entrepreneur.isEmpty()) { |  | ||||||
|             logger.error("updateEntrepreneur : No entrepreneur found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cet entrepreneur n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (school != null) { |  | ||||||
|             entrepreneur.get().setSchool(school); |  | ||||||
|         } |  | ||||||
|         if (course != null) { |  | ||||||
|             entrepreneur.get().setCourse(course); |  | ||||||
|         } |  | ||||||
|         if (sneeStatus != null) { |  | ||||||
|             entrepreneur.get().setSneeStatus(sneeStatus); |  | ||||||
|         } |  | ||||||
|         return this.entrepreneurRepository.save(entrepreneur.get()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Entrepreneur> GetEntrepreneurByProject(Project project) { |  | ||||||
|         return this.entrepreneurRepository.getEntrepreneurByProjectParticipation(project); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,124 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import static enseirb.myinpulse.model.ProjectDecisionValue.PENDING; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Administrator; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.ProjectDecisionValue; |  | ||||||
| import enseirb.myinpulse.repository.ProjectRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.time.LocalDate; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class ProjectService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final ProjectRepository projectRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     ProjectService(ProjectRepository projectRepository) { |  | ||||||
|         this.projectRepository = projectRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Project> getAllProjects() { |  | ||||||
|         return this.projectRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project getProjectById(Long id) { |  | ||||||
|         Optional<Project> project = this.projectRepository.findById(id); |  | ||||||
|         if (project.isEmpty()) { |  | ||||||
|             logger.error("No project found with id {}", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce projet n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return project.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Project> getProjectsByAdminId(Administrator administrator) { |  | ||||||
|         return this.projectRepository.findByProjectAdministrator(administrator); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: validation |  | ||||||
|     public Project addNewProject(Project project) { |  | ||||||
|         return this.projectRepository.save(project); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project updateProject( |  | ||||||
|             Long id, |  | ||||||
|             String projectName, |  | ||||||
|             byte[] logo, |  | ||||||
|             LocalDate creationDate, |  | ||||||
|             ProjectDecisionValue projectStatus, |  | ||||||
|             Administrator administrator) { |  | ||||||
|         Optional<Project> project = this.projectRepository.findById(id); |  | ||||||
|  |  | ||||||
|         if (project.isEmpty()) { |  | ||||||
|             logger.error("Project with id {} not found.", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce projet n'existe pas"); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (projectName != null) { |  | ||||||
|             project.get().setProjectName(projectName); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (logo != null) { |  | ||||||
|             project.get().setLogo(logo); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (creationDate != null) { |  | ||||||
|             project.get().setCreationDate(creationDate); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (projectStatus != null) { |  | ||||||
|             // TODO: check if this is really useful |  | ||||||
|             /* |  | ||||||
|             if (!validateStatus(projectStatus)) { |  | ||||||
|                 logger.error("updateProjectStatus: Invalid status {}", projectStatus); |  | ||||||
|                 throw new ResponseStatusException( |  | ||||||
|                         HttpStatus.NOT_ACCEPTABLE, "Ce status n'est pas accepté"); |  | ||||||
|             } |  | ||||||
|             */ |  | ||||||
|             project.get().setProjectStatus(projectStatus); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (administrator != null) { |  | ||||||
|             project.get().setProjectAdministrator(administrator); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return this.projectRepository.save(project.get()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Boolean validateStatus(String status) { |  | ||||||
|         return List.of("PENDING", "ACTIVE", "ENDED").contains(status); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Project> getPendingProjects() { |  | ||||||
|         return this.projectRepository.findByProjectStatus(PENDING); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void deleteProjectById(Long id) { |  | ||||||
|         this.projectRepository.deleteById(id); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project getProjectByName(String name, boolean noerror) { |  | ||||||
|         Optional<Project> project = this.projectRepository.findByProjectName(name); |  | ||||||
|         if (project.isEmpty()) { |  | ||||||
|             if (noerror) logger.error("No project found with name {}", name); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce projet n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return project.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Project getProjectByName(String name) { |  | ||||||
|         return getProjectByName(name, false); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,60 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Report; |  | ||||||
| import enseirb.myinpulse.repository.ReportRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class ReportService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final ReportRepository reportRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     ReportService(ReportRepository reportRepository) { |  | ||||||
|         this.reportRepository = reportRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<Report> getAllReports() { |  | ||||||
|         return this.reportRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Report getReportById(Long id) { |  | ||||||
|         Optional<Report> report = this.reportRepository.findById(id); |  | ||||||
|         if (report.isEmpty()) { |  | ||||||
|             logger.error("getReportById : No report found with id {}", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce compte rendu n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return report.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: do some validation |  | ||||||
|     public Report addNewReport(Report report) { |  | ||||||
|         return this.reportRepository.save(report); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void deleteReportById(Long id) { |  | ||||||
|         this.reportRepository.deleteById(id); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Report updateReport(Long id, String reportContent) { |  | ||||||
|         Optional<Report> report = this.reportRepository.findById(id); |  | ||||||
|         if (report.isEmpty()) { |  | ||||||
|             logger.error("updateReport : No report found with id {}", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Ce compte rendu n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (reportContent != null) { |  | ||||||
|             report.get().setReportContent(reportContent); |  | ||||||
|         } |  | ||||||
|         return this.reportRepository.save(report.get()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,93 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Appointment; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.SectionCell; |  | ||||||
| import enseirb.myinpulse.repository.SectionCellRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.time.LocalDateTime; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class SectionCellService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final SectionCellRepository sectionCellRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     SectionCellService(SectionCellRepository sectionCellRepository) { |  | ||||||
|         this.sectionCellRepository = sectionCellRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<SectionCell> getAllSectionCells() { |  | ||||||
|         return this.sectionCellRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public SectionCell getSectionCellById(Long id) { |  | ||||||
|         Optional<SectionCell> sectionCell = this.sectionCellRepository.findById(id); |  | ||||||
|         if (sectionCell.isEmpty()) { |  | ||||||
|             logger.error("getSectionCellById : No sectionCell found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return sectionCell.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public SectionCell addNewSectionCell(SectionCell sectionCell) { |  | ||||||
|         return this.sectionCellRepository.save(sectionCell); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void removeSectionCellById(Long id) { |  | ||||||
|         this.sectionCellRepository.deleteById(id); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public SectionCell updateSectionCell( |  | ||||||
|             Long id, Long sectionId, String contentSectionCell, LocalDateTime modificationDate) { |  | ||||||
|         Optional<SectionCell> sectionCell = this.sectionCellRepository.findById(id); |  | ||||||
|         if (sectionCell.isEmpty()) { |  | ||||||
|             logger.error("updateSectionCell : No sectionCell found with id {}", id); |  | ||||||
|             throw new ResponseStatusException( |  | ||||||
|                     HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (sectionId != null) { |  | ||||||
|             sectionCell.get().setSectionId(sectionId); |  | ||||||
|         } |  | ||||||
|         if (contentSectionCell != null) { |  | ||||||
|             sectionCell.get().setContentSectionCell(contentSectionCell); |  | ||||||
|         } |  | ||||||
|         if (modificationDate != null) { |  | ||||||
|             sectionCell.get().setModificationDate(modificationDate); |  | ||||||
|         } |  | ||||||
|         return this.sectionCellRepository.save(sectionCell.get()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<SectionCell> getSectionCellsByProject(Project project, Long sectionId) { |  | ||||||
|         return this.sectionCellRepository.findByProjectSectionCellAndSectionId(project, sectionId); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Long getProjectId(Long sectionCellId) { |  | ||||||
|         SectionCell sectionCell = getSectionCellById(sectionCellId); |  | ||||||
|         Project sectionProject = sectionCell.getProjectSectionCell(); |  | ||||||
|         return sectionProject.getIdProject(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<Appointment> getAppointmentsBySectionCellId(Long sectionCellId) { |  | ||||||
|         SectionCell sectionCell = getSectionCellById(sectionCellId); |  | ||||||
|         return sectionCell.getAppointmentSectionCell(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<SectionCell> getSectionCellsByProjectAndSectionIdBeforeDate( |  | ||||||
|             Project project, long sectionId, LocalDateTime date) { |  | ||||||
|         return sectionCellRepository.findByProjectSectionCellAndSectionIdAndModificationDateBefore( |  | ||||||
|                 project, sectionId, date); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,81 +0,0 @@ | |||||||
| package enseirb.myinpulse.service.database; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.User; |  | ||||||
| import enseirb.myinpulse.repository.UserRepository; |  | ||||||
|  |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.http.HttpStatus; |  | ||||||
| import org.springframework.stereotype.Service; |  | ||||||
| import org.springframework.web.bind.annotation.PathVariable; |  | ||||||
| import org.springframework.web.bind.annotation.RequestBody; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.util.Optional; |  | ||||||
|  |  | ||||||
| @Service |  | ||||||
| public class UserService { |  | ||||||
|  |  | ||||||
|     protected static final Logger logger = LogManager.getLogger(); |  | ||||||
|  |  | ||||||
|     private final UserRepository userRepository; |  | ||||||
|  |  | ||||||
|     @Autowired |  | ||||||
|     UserService(UserRepository userRepository) { |  | ||||||
|         this.userRepository = userRepository; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<User> getAllUsers() { |  | ||||||
|         return this.userRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO |  | ||||||
|     public User getUserByEmail(String email) { |  | ||||||
|         Optional<User> opt_user = this.userRepository.findByPrimaryMail(email); |  | ||||||
|  |  | ||||||
|         if (opt_user.isEmpty()) { |  | ||||||
|             logger.error("getUserByEmail : No user found with email {}", email); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Cet utilisateur n'existe pas"); |  | ||||||
|         } |  | ||||||
|         return opt_user.get(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Iterable<User> allUsers() { |  | ||||||
|         return this.userRepository.findAll(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public User addUser(@RequestBody User user) { |  | ||||||
|         return this.userRepository.save(user); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public User updateUser( |  | ||||||
|             @PathVariable Long id, |  | ||||||
|             String userSurname, |  | ||||||
|             String userName, |  | ||||||
|             String primaryMail, |  | ||||||
|             String secondaryMail, |  | ||||||
|             String phoneNumber) { |  | ||||||
|         Optional<User> user = userRepository.findById(id); |  | ||||||
|         if (user.isEmpty()) { |  | ||||||
|             logger.error("updateUser : No user found with id {}", id); |  | ||||||
|             throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Cet utilisateur n'existe pas"); |  | ||||||
|         } |  | ||||||
|         if (userName != null) { |  | ||||||
|             user.get().setUserName(userName); |  | ||||||
|         } |  | ||||||
|         if (userSurname != null) { |  | ||||||
|             user.get().setUserSurname(userSurname); |  | ||||||
|         } |  | ||||||
|         if (primaryMail != null) { |  | ||||||
|             user.get().setPrimaryMail(primaryMail); |  | ||||||
|         } |  | ||||||
|         if (secondaryMail != null) { |  | ||||||
|             user.get().setSecondaryMail(secondaryMail); |  | ||||||
|         } |  | ||||||
|         if (phoneNumber != null) { |  | ||||||
|             user.get().setPhoneNumber(phoneNumber); |  | ||||||
|         } |  | ||||||
|         return this.userRepository.save(user.get()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,8 +1,4 @@ | |||||||
| spring.application.name=myinpulse | spring.application.name=myinpulse | ||||||
| spring.security.oauth2.resourceserver.jwt.jwk-set-uri=http://localhost:7080/realms/test/protocol/openid-connect/certs | spring.security.oauth2.resourceserver.jwt.jwk-set-uri=http://localhost:7080/realms/test/protocol/openid-connect/certs | ||||||
| spring.security.oauth2.resourceserver.jwt.issuer-uri=http://localhost:7080/realms/test | spring.security.oauth2.resourceserver.jwt.issuer-uri=http://localhost:7080/realms/test | ||||||
| spring.datasource.url=jdbc:postgresql://${DATABASE_URL}/${BACKEND_DB} | logging.level.org.springframework.security=DEBUG | ||||||
| spring.datasource.username=${BACKEND_USER} |  | ||||||
| spring.datasource.password=${BACKEND_PASSWORD} |  | ||||||
| spring.jpa.hibernate.ddl-auto=update |  | ||||||
| logging.pattern.console=%d{yyyy-MMM-dd HH:mm:ss.SSS} [%thread] %highlight(%-5level) %cyan(%logger{15}) - %msg %n |  | ||||||
|   | |||||||
| @@ -1,99 +0,0 @@ | |||||||
| TRUNCATE project, user_inpulse, entrepreneur, administrator, section_cell, appointment, report, annotation CASCADE; |  | ||||||
|  |  | ||||||
| SELECT setval('annotation_id_annotation_seq', 1, false); |  | ||||||
| SELECT setval('appointment_id_appointment_seq', 1, false); |  | ||||||
| SELECT setval('make_appointment_id_make_appointment_seq', 1, false); |  | ||||||
| SELECT setval('project_id_project_seq', 1, false); |  | ||||||
| SELECT setval('report_id_report_seq', 1, false); |  | ||||||
| SELECT setval('section_cell_id_section_cell_seq', 1, false); |  | ||||||
| SELECT setval('user_inpulse_id_user_seq', 1, false); |  | ||||||
|  |  | ||||||
| INSERT INTO user_inpulse (user_surname, user_name, primary_mail, secondary_mail, phone_number) |  | ||||||
| VALUES ('Dupont', 'Dupond', 'super@mail.fr', 'super2@mail.fr', '06 45 72 45 98'), |  | ||||||
|        ('Martin', 'Matin', 'genial@mail.fr', 'genial2@mail.fr', '06 52 14 58 73'), |  | ||||||
|        ('Charvet', 'Lautre', 'mieux@tmail.fr', 'mieux2@tmail.fr', '07 49 82 16 35'), |  | ||||||
|        ('Leguez', 'Theo', 'bof@mesmails.fr', 'bof2@mesmails.fr', '+33 6 78 14 25 29'), |  | ||||||
|        ('Kia', 'Bi', 'special@mail.fr', 'special2@mail.fr', '07 65 31 38 95'), |  | ||||||
|        ('Ducaillou', 'Pierre', 'maildefou@xyz.fr', 'maildefou2@xyz.fr', '06 54 78 12 62'), |  | ||||||
|        ('Janine', 'Dave', 'janine@labri.fr', 'janine2@labri.fr', '06 87 12 45 95'); |  | ||||||
|  |  | ||||||
| INSERT INTO administrator (id_administrator) |  | ||||||
| VALUES (7); |  | ||||||
|  |  | ||||||
| INSERT INTO project (project_name, logo, creation_date, project_status, id_administrator) |  | ||||||
| VALUES ('Eau du robinet', decode('013d7d16d7ad4fefb61bd95b765c8ceb', 'hex'), TO_DATE('01-OCT-2023', 'DD-MON-YYYY'), |  | ||||||
|         'En cours', 7), |  | ||||||
|        ('Air oxygéné', decode('150647a0984e8f228cd14b54', 'hex'), TO_DATE('04-APR-2024', 'DD-MON-YYYY'), 'En cours', 7), |  | ||||||
|        ('Débat concours', decode('022024abd5486e245c145dda65116f', 'hex'), TO_DATE('22-NOV-2023', 'DD-MON-YYYY'), |  | ||||||
|         'Suspendu', 7), |  | ||||||
|        ('HDeirbMI', decode('ab548d6c1d595a2975e6476f544d14c55a', 'hex'), TO_DATE('07-DEC-2024', 'DD-MON-YYYY'), |  | ||||||
|         'Lancement', 7); |  | ||||||
|  |  | ||||||
|  |  | ||||||
| INSERT INTO entrepreneur (school, course, snee_status, id_entrepreneur, id_project_participation, id_project_proposed) |  | ||||||
| VALUES ('ENSEIRB-MATMECA', 'INFO', TRUE, 1, 4, 4), |  | ||||||
|        ('ENSC', 'Cognitique', TRUE, 2, 2, null), |  | ||||||
|        ('ENSEIRB-MATMECA', 'MATMECA', FALSE, 3, 3, 3), |  | ||||||
|        ('SupOptique', 'Classique', TRUE, 4, 1, 1), |  | ||||||
|        ('ENSEGID', 'Géoscience', FALSE, 5, 1, null), |  | ||||||
|        ('ENSMAC', 'Matériaux composites - Mécanique', FALSE, 6, 2, 2); |  | ||||||
|  |  | ||||||
|  |  | ||||||
| INSERT INTO section_cell (title, content_section_cell, modification_date, id_project) |  | ||||||
| VALUES ('Problème', 'les problèmes...', TO_TIMESTAMP('15-JAN-2025 09:30:20', 'DD-MON-YYYY, HH24:MI:SS'), 2), |  | ||||||
|        ('Segment de client', 'Le segment AB passant le client n°8 est de longueur 32mm. |  | ||||||
|     Le segment BC a quant à lui un longueur de 28mm. Quelle la longueur du segment AC ?', |  | ||||||
|         TO_TIMESTAMP('12-OCT-2022 17:47:38', 'DD-MON-YYYY, HH24:MI:SS'), 3), |  | ||||||
|        ('Proposition de valeur unique', '''Son prix est de 2594€'' ''Ah oui c''est unique en effet', |  | ||||||
|         TO_TIMESTAMP('25-MAY-2024 11:12:04', 'DD-MON-YYYY, HH24:MI:SS'), 2), |  | ||||||
|        ('Solution', 'Un problème ? Une solution', TO_TIMESTAMP('08-FEB-2024 10:17:53', 'DD-MON-YYYY, HH24:MI:SS'), 1), |  | ||||||
|        ('Canaux', 'Ici nous avons la Seine, là-bas le Rhin, oh et plus loin le canal de Suez', |  | ||||||
|         TO_TIMESTAMP('19-JUL-2023 19:22:45', 'DD-MON-YYYY, HH24:MI:SS'), 4), |  | ||||||
|        ('Sources de revenus', 'Y''en n''a pas on est pas payé. Enfin y''a du café quoi', |  | ||||||
|         TO_TIMESTAMP('12-JAN-2025 11:40:26', 'DD-MON-YYYY, HH24:MI:SS'), 1), |  | ||||||
|        ('Structure des coûts', '''Ah oui là ça va faire au moins 1000€ par mois'', Eirbware', |  | ||||||
|         TO_TIMESTAMP('06-FEB-2025 13:04:06', 'DD-MON-YYYY, HH24:MI:SS'), 3), |  | ||||||
|        ('Indicateurs clés', 'On apprend les clés comme des badges, ça se fait', |  | ||||||
|         TO_TIMESTAMP('05-FEB-2025 12:42:38', 'DD-MON-YYYY, HH24:MI:SS'), 4), |  | ||||||
|        ('Avantages concurrentiel', 'On est meilleur', TO_TIMESTAMP('23-APR-2024 16:24:02', 'DD-MON-YYYY, HH24:MI:SS'), |  | ||||||
|         2); |  | ||||||
|  |  | ||||||
| INSERT INTO appointment (appointment_date, appointment_time, appointment_duration, appointment_place, |  | ||||||
|                          appointment_subject) |  | ||||||
| VALUES (TO_DATE('24-DEC-2023', 'DD-MON-YYYY'), '00:00:00', '00:37:53', 'À la maison', 'Ouvrir les cadeaux'), |  | ||||||
|        (TO_DATE('15-AUG-2024', 'DD-MON-YYYY'), '22:35:00', '00:12:36', 'Sur les quais ou dans un champ probablement', |  | ||||||
|         'BOUM BOUM les feux d''artifices (on fête quoi déjà ?)'), |  | ||||||
|        (TO_DATE('28-FEB-2023', 'DD-MON-YYYY'), '14:20:00', '00:20:00', 'Salle TD 15', |  | ||||||
|         'Ah mince c''est pas une année bissextile !'), |  | ||||||
|        (TO_DATE('23-JAN-2024', 'DD-MON-YYYY'), '12:56:27', '11:03:33', 'Là où le vent nous porte', |  | ||||||
|         'Journée la plus importante de l''année'), |  | ||||||
|        (TO_DATE('25-AUG-2025', 'DD-MON-YYYY'), '00:09:00', '01:00:00', 'Euh c''est par où l''amphi 56 ?', |  | ||||||
|         'Rentrée scolaire (il fait trop froid c''est quoi ça on est en août)'); |  | ||||||
|  |  | ||||||
| INSERT INTO report (report_content, id_appointment) |  | ||||||
| VALUES ('Ah oui ça c''est super, ah ouais j''aime bien, bien vu de penser à ça', 1), |  | ||||||
|        ('Bonne réunion', 3), |  | ||||||
|        ('Ouais, j''ai rien compris mais niquel on fait comme vous avez dit', 3), |  | ||||||
|        ('Non non ça va pas du tout ce que tu me proposes, faut tout refaire', 4), |  | ||||||
|        ('Réponse de la DSI : non', 2), |  | ||||||
|        ('Trop dommage qu''Apple ait sorti leur logiciel avant nous, on avait la même idée et tout on aurait tellement pu leur faire de la concurrence', |  | ||||||
|         5); |  | ||||||
|  |  | ||||||
| INSERT INTO annotation (comment, id_administrator, id_section_cell) |  | ||||||
| VALUES ('faut changer ça hein', 7, 5), |  | ||||||
|        ('??? sérieusement, vous pensez que c''est une bonne idée ?', 7, 7), |  | ||||||
|        ('ok donc ça c''est votre business plan, bah glhf la team', 7, 2); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| DROP TABLE IF EXISTS administrateurs, projets, utilisateurs, entrepreneurs, sections, rendez_vous, comptes_rendus, concerner CASCADE; |  | ||||||
| DROP TABLE IF EXISTS administrator, project, user_inpulse, entrepreneur, section_cell, appointment, make_appointment, report, annotation, concern CASCADE; |  | ||||||
| @@ -1,224 +0,0 @@ | |||||||
| package enseirb.myinpulse; |  | ||||||
|  |  | ||||||
| import static enseirb.myinpulse.model.ProjectDecisionValue.*; |  | ||||||
|  |  | ||||||
| import static org.junit.jupiter.api.Assertions.*; |  | ||||||
|  |  | ||||||
| import enseirb.myinpulse.model.Administrator; |  | ||||||
| import enseirb.myinpulse.model.Entrepreneur; |  | ||||||
| import enseirb.myinpulse.model.Project; |  | ||||||
| import enseirb.myinpulse.model.ProjectDecision; |  | ||||||
| import enseirb.myinpulse.service.AdminApiService; |  | ||||||
| import enseirb.myinpulse.service.database.AdministratorService; |  | ||||||
| import enseirb.myinpulse.service.database.EntrepreneurService; |  | ||||||
| import enseirb.myinpulse.service.database.ProjectService; |  | ||||||
|  |  | ||||||
| import org.junit.jupiter.api.BeforeAll; |  | ||||||
| import org.junit.jupiter.api.Test; |  | ||||||
| import org.springframework.beans.factory.annotation.Autowired; |  | ||||||
| import org.springframework.boot.test.context.SpringBootTest; |  | ||||||
| import org.springframework.transaction.annotation.Transactional; |  | ||||||
| import org.springframework.web.server.ResponseStatusException; |  | ||||||
|  |  | ||||||
| import java.time.LocalDate; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| @SpringBootTest |  | ||||||
| @Transactional |  | ||||||
| public class AdminApiServiceTest { |  | ||||||
|     private static long administratorid; |  | ||||||
|     private static Administrator administrator; |  | ||||||
|     private static Entrepreneur entrepreneur; |  | ||||||
|     @Autowired private AdminApiService adminApiService; |  | ||||||
|     @Autowired private ProjectService projectService; |  | ||||||
|  |  | ||||||
|     @BeforeAll |  | ||||||
|     static void setup( |  | ||||||
|             @Autowired AdministratorService administratorService, |  | ||||||
|             @Autowired ProjectService projectService, |  | ||||||
|             @Autowired EntrepreneurService entrepreneurService) { |  | ||||||
|         administratorService.addAdministrator( |  | ||||||
|                 new Administrator( |  | ||||||
|                         "admin", |  | ||||||
|                         "admin", |  | ||||||
|                         "testAdminEmpty@example.com", |  | ||||||
|                         "testAdmin@example.com", |  | ||||||
|                         "")); |  | ||||||
|         administrator = |  | ||||||
|                 administratorService.addAdministrator( |  | ||||||
|                         new Administrator( |  | ||||||
|                                 "admin2", |  | ||||||
|                                 "admin2", |  | ||||||
|                                 "testAdminFull@example.com", |  | ||||||
|                                 "testAdmin@example.com", |  | ||||||
|                                 "")); |  | ||||||
|         administratorid = administrator.getIdUser(); |  | ||||||
|         entrepreneur = |  | ||||||
|                 new Entrepreneur( |  | ||||||
|                         "JeSuisUnEntrepreneurDeCompet", |  | ||||||
|                         "EtUé", |  | ||||||
|                         "Entrepreneur@inpulse.com", |  | ||||||
|                         "mail2", |  | ||||||
|                         "phone", |  | ||||||
|                         "Ensimag    nan jdeconne ENSEIRB (-matmeca mais on s'en fout)", |  | ||||||
|                         "info ofc", |  | ||||||
|                         false); |  | ||||||
|         entrepreneurService.addEntrepreneur(entrepreneur); |  | ||||||
|         projectService.addNewProject( |  | ||||||
|                 new Project( |  | ||||||
|                         "sampleProjectAdminApiService", |  | ||||||
|                         null, |  | ||||||
|                         LocalDate.now(), |  | ||||||
|                         ACTIVE, |  | ||||||
|                         administratorService.getAdministratorByPrimaryMain( |  | ||||||
|                                 "testAdminFull@example.com"))); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private <T> List<T> IterableToList(Iterable<T> iterable) { |  | ||||||
|         List<T> l = new ArrayList<>(); |  | ||||||
|         iterable.forEach(l::add); |  | ||||||
|         return l; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void getProjectOfAdminIsEmpty() { |  | ||||||
|         Iterable<Project> projects = |  | ||||||
|                 adminApiService.getProjectsOfAdmin("testAdminEmpty@example.com"); |  | ||||||
|         assertEquals(0, IterableToList(projects).size()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void getProjectOfInexistantAdminFails() { |  | ||||||
|         String nonExistentAdminEmail = "testInexistantAdmin@example.com"; |  | ||||||
|  |  | ||||||
|         assertThrows( |  | ||||||
|                 ResponseStatusException.class, |  | ||||||
|                 () -> { |  | ||||||
|                     adminApiService.getProjectsOfAdmin(nonExistentAdminEmail); |  | ||||||
|                 }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void getProjectOfAdminNotEmpty() { |  | ||||||
|         Iterable<Project> projects = |  | ||||||
|                 adminApiService.getProjectsOfAdmin("testAdminFull@example.com"); |  | ||||||
|         List<Project> l = IterableToList(projects); |  | ||||||
|         assertEquals(1, l.size()); |  | ||||||
|         Project p = l.getFirst(); |  | ||||||
|         assertEquals(p.getProjectName(), "sampleProjectAdminApiService"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void getPendingProjectsEmpty() { |  | ||||||
|         assertEquals(0, IterableToList(this.adminApiService.getPendingProjects()).size()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void getPendingProjectsNotEmpty() { |  | ||||||
|         this.projectService.addNewProject( |  | ||||||
|                 new Project( |  | ||||||
|                         "PendingProjectAdminApiService1", null, LocalDate.now(), PENDING, null)); |  | ||||||
|         this.projectService.addNewProject( |  | ||||||
|                 new Project( |  | ||||||
|                         "PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null)); |  | ||||||
|         Iterable<Project> pendingProjects = this.adminApiService.getPendingProjects(); |  | ||||||
|         List<Project> pendingProjectsList = IterableToList(pendingProjects); |  | ||||||
|         assertEquals(2, pendingProjectsList.size()); |  | ||||||
|         assertTrue( |  | ||||||
|                 List.of("PendingProjectAdminApiService1", "PendingProjectAdminApiService2") |  | ||||||
|                         .contains(pendingProjectsList.getFirst().getProjectName())); |  | ||||||
|         assertTrue( |  | ||||||
|                 List.of("PendingProjectAdminApiService1", "PendingProjectAdminApiService2") |  | ||||||
|                         .contains(pendingProjectsList.getLast().getProjectName())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void validateInexistantProject() { |  | ||||||
|         ProjectDecision d = new ProjectDecision(-1, 0, 1); |  | ||||||
|         assertThrows(ResponseStatusException.class, () -> this.adminApiService.validateProject(d)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void validateExistantProject() { |  | ||||||
|         Project p = |  | ||||||
|                 new Project("PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null); |  | ||||||
|         this.projectService.addNewProject(p); |  | ||||||
|         assertEquals(PENDING, p.getProjectStatus()); |  | ||||||
|         ProjectDecision d = new ProjectDecision(p.getIdProject(), administratorid, 1); |  | ||||||
|         this.adminApiService.validateProject(d); |  | ||||||
|         assertEquals(ACTIVE, p.getProjectStatus()); |  | ||||||
|  |  | ||||||
|         // Check if the project was really updated in the database |  | ||||||
|         assertEquals(0, IterableToList(this.adminApiService.getPendingProjects()).size()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void refuseExistantProject() { |  | ||||||
|         Project p = |  | ||||||
|                 new Project("PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null); |  | ||||||
|         this.projectService.addNewProject(p); |  | ||||||
|         assertEquals(PENDING, p.getProjectStatus()); |  | ||||||
|         ProjectDecision d = new ProjectDecision(p.getIdProject(), administratorid, 0); |  | ||||||
|         this.adminApiService.validateProject(d); |  | ||||||
|         assertEquals(REJECTED, p.getProjectStatus()); |  | ||||||
|  |  | ||||||
|         // Check if the project was really updated in the database |  | ||||||
|         assertEquals(0, IterableToList(this.adminApiService.getPendingProjects()).size()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void addProject() { |  | ||||||
|         assertEquals(0, IterableToList(this.adminApiService.getPendingProjects()).size()); |  | ||||||
|         Project p1 = |  | ||||||
|                 new Project("PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null); |  | ||||||
|         this.adminApiService.addNewProject(p1); |  | ||||||
|  |  | ||||||
|         assertEquals(1, IterableToList(this.adminApiService.getPendingProjects()).size()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void addProjectToAdmin() { |  | ||||||
|         assertEquals(0, administrator.getListProject().size()); |  | ||||||
|         Project p1 = new Project("assProjectToAdmin", null, LocalDate.now(), ACTIVE, administrator); |  | ||||||
|         this.adminApiService.addNewProject(p1); |  | ||||||
|         assertEquals(1, administrator.getListProject().size()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void addProjectToUser() { |  | ||||||
|         assertNull(entrepreneur.getProjectParticipation()); |  | ||||||
|         Project p1 = |  | ||||||
|                 new Project("assProjectToAdmin", null, LocalDate.now(), ACTIVE, null, entrepreneur); |  | ||||||
|         this.adminApiService.addNewProject(p1); |  | ||||||
|         assertEquals(p1, entrepreneur.getProjectParticipation()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void addProjectWithManyUsers() { |  | ||||||
|         Entrepreneur e1 = new Entrepreneur(); |  | ||||||
|         Entrepreneur e2 = new Entrepreneur(); |  | ||||||
|         Entrepreneur e3 = new Entrepreneur(); |  | ||||||
|         assertNull(e1.getProjectParticipation()); |  | ||||||
|         assertNull(e2.getProjectParticipation()); |  | ||||||
|         assertNull(e3.getProjectParticipation()); |  | ||||||
|         Project p1 = new Project("assProjectToAdmin", null, LocalDate.now(), ACTIVE, null, null); |  | ||||||
|         p1.updateListEntrepreneurParticipation(e1); |  | ||||||
|         p1.updateListEntrepreneurParticipation(e2); |  | ||||||
|         p1.updateListEntrepreneurParticipation(e3); |  | ||||||
|         this.adminApiService.addNewProject(p1); |  | ||||||
|         assertEquals(p1, e1.getProjectParticipation()); |  | ||||||
|         assertEquals(p1, e2.getProjectParticipation()); |  | ||||||
|         assertEquals(p1, e3.getProjectParticipation()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     void addDuplicateProject() { |  | ||||||
|         Project p1 = |  | ||||||
|                 new Project("PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null); |  | ||||||
|         Project p2 = |  | ||||||
|                 new Project("PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null); |  | ||||||
|         this.adminApiService.addNewProject(p1); |  | ||||||
|         assertThrows(ResponseStatusException.class, () -> this.adminApiService.addNewProject(p2)); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,13 +1,13 @@ | |||||||
| package enseirb.myinpulse; | package enseirb.myinpulse; | ||||||
|  |  | ||||||
| import org.junit.jupiter.api.DisplayName; |  | ||||||
| import org.junit.jupiter.api.Test; | import org.junit.jupiter.api.Test; | ||||||
| import org.springframework.boot.test.context.SpringBootTest; | import org.springframework.boot.test.context.SpringBootTest; | ||||||
|  |  | ||||||
| @SpringBootTest | @SpringBootTest | ||||||
| class MyinpulseApplicationTests { | class MyinpulseApplicationTests { | ||||||
|  |  | ||||||
|     @Test | 	@Test | ||||||
|     @DisplayName("contextLoad => Test if the context can load, i.e. the application can start") | 	void contextLoads() { | ||||||
|     void contextLoads() {} | 	} | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +0,0 @@ | |||||||
| spring.datasource.driver-class-name=org.h2.Driver |  | ||||||
| spring.datasource.url=jdbc:h2:mem:db;DB_CLOSE_DELAY=-1 |  | ||||||
| spring.datasource.username=sa |  | ||||||
| spring.datasource.password=sa |  | ||||||
| spring.sql.init.mode=never |  | ||||||
| spring.application.name=myinpulse-test |  | ||||||
| spring.security.oauth2.resourceserver.jwt.jwk-set-uri=http://localhost:7080/realms/test/protocol/openid-connect/certs |  | ||||||
| spring.security.oauth2.resourceserver.jwt.issuer-uri=http://localhost:7080/realms/test |  | ||||||
| spring.jpa.hibernate.ddl-auto=update |  | ||||||
| logging.pattern.console=%d{yyyy-MMM-dd HH:mm:ss.SSS} [%thread] %highlight(%-5level) %cyan(%logger{15}) - %msg %n |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| POSTGRES_DB=postgres_db |  | ||||||
| POSTGRES_USER=postgres |  | ||||||
| POSTGRES_PASSWORD=postgres_db_user_password |  | ||||||
|  |  | ||||||
| KEYCLOAK_ADMIN=admin |  | ||||||
| KEYCLOAK_ADMIN_PASSWORD=admin |  | ||||||
| KEYCLOAK_HOSTNAME=localhost |  | ||||||
| KEYCLOAK_DB=keycloak_db |  | ||||||
| KEYCLOAK_USER=keycloak_db_user |  | ||||||
| KEYCLOAK_PASSWORD=keycloak_db_user_password |  | ||||||
|  |  | ||||||
| BACKEND_DB=backend_db |  | ||||||
| BACKEND_USER=backend_db_user |  | ||||||
| BACKEND_PASSWORD=backend_db_user_password |  | ||||||
|  |  | ||||||
| DATABASE_URL=localhost:5433 |  | ||||||
|  |  | ||||||
| VITE_KEYCLOAK_URL=http://localhost:7080 |  | ||||||
| VITE_KEYCLOAK_CLIENT_ID=myinpulse-dev |  | ||||||
| VITE_KEYCLOAK_REALM=test |  | ||||||
| VITE_APP_URL=http://localhost:5173 |  | ||||||
| VITE_BACKEND_URL=http://localhost:8081/ |  | ||||||
| @@ -1,14 +1,15 @@ | |||||||
| services: | services: | ||||||
|   postgres: |   postgres: | ||||||
|     env_file: .env |     image: postgres:latest | ||||||
|     build: |  | ||||||
|       context: postgres/ |  | ||||||
|       dockerfile: Dockerfile |  | ||||||
|     container_name: MyINPulse-DB |     container_name: MyINPulse-DB | ||||||
|     ports: |     #ports: | ||||||
|       - 5433:5432 |     #  - 5432:5432 | ||||||
|     volumes: |     volumes: | ||||||
|       - ./postgres/data:/var/lib/postgresql/data |       - ./postgres:/var/lib/postgresql/data | ||||||
|  |     environment: | ||||||
|  |       POSTGRES_DB: ${POSTGRES_DB} | ||||||
|  |       POSTGRES_USER: ${POSTGRES_USER} | ||||||
|  |       POSTGRES_PASSWORD: ${POSTGRES_PASSWORD} | ||||||
|  |  | ||||||
|   keycloak: |   keycloak: | ||||||
|     container_name: MyINPulse-keycloak |     container_name: MyINPulse-keycloak | ||||||
|   | |||||||
| @@ -1,22 +0,0 @@ | |||||||
| POSTGRES_DB=postgres_db |  | ||||||
| POSTGRES_USER=postgres |  | ||||||
| POSTGRES_PASSWORD=postgres_db_user_password |  | ||||||
|  |  | ||||||
| KEYCLOAK_ADMIN=admin |  | ||||||
| KEYCLOAK_ADMIN_PASSWORD=admin |  | ||||||
| KEYCLOAK_HOSTNAME=localhost |  | ||||||
| KEYCLOAK_DB=keycloak_db |  | ||||||
| KEYCLOAK_USER=keycloak_db_user |  | ||||||
| KEYCLOAK_PASSWORD=keycloak_db_user_password |  | ||||||
|  |  | ||||||
| BACKEND_DB=backend_db |  | ||||||
| BACKEND_USER=backend_db_user |  | ||||||
| BACKEND_PASSWORD=backend_db_user_password |  | ||||||
|  |  | ||||||
| DATABASE_URL=localhost:5433 |  | ||||||
|  |  | ||||||
| VITE_KEYCLOAK_URL=http://localhost:7080 |  | ||||||
| VITE_KEYCLOAK_CLIENT_ID=myinpulse |  | ||||||
| VITE_KEYCLOAK_REALM=test |  | ||||||
| VITE_APP_URL=http://localhost:8080 |  | ||||||
| VITE_BACKEND_URL=http://localhost:8081/ |  | ||||||
							
								
								
									
										5
									
								
								config/backdev.front.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								config/backdev.front.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | VITE_KEYCLOAK_URL=http://localhost:7080 | ||||||
|  | VITE_KEYCLOAK_CLIENT_ID=myinpulse | ||||||
|  | VITE_KEYCLOAK_REALM=test | ||||||
|  | VITE_APP_URL=http://localhost:8080 | ||||||
|  | VITE_BACKEND_URL=http://localhost:8081/ | ||||||
							
								
								
									
										6
									
								
								config/backdev.main.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								config/backdev.main.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | POSTGRES_DB=keycloak_db | ||||||
|  | POSTGRES_USER=keycloak_db_user | ||||||
|  | POSTGRES_PASSWORD=keycloak_db_user_password | ||||||
|  | KEYCLOAK_ADMIN=admin | ||||||
|  | KEYCLOAK_ADMIN_PASSWORD=admin | ||||||
|  | KEYCLOAK_HOSTNAME=localhost | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| services: |  | ||||||
|   postgres: |  | ||||||
|     env_file: .env |  | ||||||
|     build: |  | ||||||
|       context: postgres/ |  | ||||||
|       dockerfile: Dockerfile |  | ||||||
|     container_name: MyINPulse-DB |  | ||||||
|     ports: |  | ||||||
|       - 5433:5432 |  | ||||||
|     volumes: |  | ||||||
|       - ./postgres/data:/var/lib/postgresql/data |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   keycloak: |  | ||||||
|     container_name: MyINPulse-keycloak |  | ||||||
|     build: |  | ||||||
|       context: ./keycloak |  | ||||||
|       dockerfile: Dockerfile |  | ||||||
|       args: |  | ||||||
|         KC_DB: postgres |  | ||||||
|         KC_DB_URL: jdbc:postgresql://postgres/${POSTGRES_DB} |  | ||||||
|         KC_DB_USERNAME: ${POSTGRES_USER} |  | ||||||
|         KC_DB_PASSWORD: ${POSTGRES_PASSWORD} |  | ||||||
|     environment: |  | ||||||
|       KC_HOSTNAME_PORT: 7080 |  | ||||||
|       KC_HOSTNAME_STRICT_BACKCHANNEL: "true" |  | ||||||
|       KC_BOOTSTRAP_ADMIN_USERNAME: ${KEYCLOAK_ADMIN} |  | ||||||
|       KC_BOOTSTRAP_ADMIN_PASSWORD: ${KEYCLOAK_ADMIN_PASSWORD} |  | ||||||
|       KC_LOG_LEVEL: info |  | ||||||
|     command: ["start-dev", "--http-port", "7080", "--https-port", "7443", "--hostname", "${KEYCLOAK_HOSTNAME}"] |  | ||||||
|     ports: |  | ||||||
|       - "7080:7080" |  | ||||||
|       - "7443:7443" |  | ||||||
|     depends_on: |  | ||||||
|       - postgres |  | ||||||
|  |  | ||||||
|   #front: |  | ||||||
|   #  build: |  | ||||||
|   #    context: ./front/ |  | ||||||
|   #    dockerfile: Dockerfile |  | ||||||
|   #  container_name: MyINPulse-front |  | ||||||
|   #  ports: |  | ||||||
|   #    - "8080:80" |  | ||||||
|  |  | ||||||
|   #back: |  | ||||||
|   #  build: |  | ||||||
|   #    context: ./MyINPulse-back/ |  | ||||||
|   #    dockerfile: Dockerfile |  | ||||||
|   #  container_name: MyINPulse-back |  | ||||||
|   #  ports: |  | ||||||
|   #    - "8081:8080" |  | ||||||
|    |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| POSTGRES_DB=postgres_db |  | ||||||
| POSTGRES_USER=postgres |  | ||||||
| POSTGRES_PASSWORD=postgres_db_user_password |  | ||||||
|  |  | ||||||
| KEYCLOAK_ADMIN=admin |  | ||||||
| KEYCLOAK_ADMIN_PASSWORD=admin |  | ||||||
| KEYCLOAK_HOSTNAME=localhost |  | ||||||
| KEYCLOAK_DB=keycloak_db |  | ||||||
| KEYCLOAK_USER=keycloak_db_user |  | ||||||
| KEYCLOAK_PASSWORD=keycloak_db_user_password |  | ||||||
|  |  | ||||||
| BACKEND_DB=backend_db |  | ||||||
| BACKEND_USER=backend_db_user |  | ||||||
| BACKEND_PASSWORD=backend_db_user_password |  | ||||||
|  |  | ||||||
| DATABASE_URL=localhost:5433 |  | ||||||
|  |  | ||||||
| VITE_KEYCLOAK_URL=http://localhost:7080 |  | ||||||
| VITE_KEYCLOAK_CLIENT_ID=myinpulse-dev |  | ||||||
| VITE_KEYCLOAK_REALM=test |  | ||||||
| VITE_APP_URL=http://localhost:5173 |  | ||||||
| VITE_BACKEND_URL=http://localhost:8081/ |  | ||||||
| @@ -1,15 +1,15 @@ | |||||||
| services: | services: | ||||||
|   postgres: |   postgres: | ||||||
|     env_file: .env |     image: postgres:latest | ||||||
|     build: |  | ||||||
|       context: postgres/ |  | ||||||
|       dockerfile: Dockerfile |  | ||||||
|     container_name: MyINPulse-DB |     container_name: MyINPulse-DB | ||||||
|     #ports: |     #ports: | ||||||
|     #  - 5432:5432 |     #  - 5432:5432 | ||||||
|     volumes: |     volumes: | ||||||
|       - ./postgres/data:/var/lib/postgresql/data |       - ./postgres:/var/lib/postgresql/data | ||||||
|  |     environment: | ||||||
|  |       POSTGRES_DB: ${POSTGRES_DB} | ||||||
|  |       POSTGRES_USER: ${POSTGRES_USER} | ||||||
|  |       POSTGRES_PASSWORD: ${POSTGRES_PASSWORD} | ||||||
|  |  | ||||||
|   keycloak: |   keycloak: | ||||||
|     container_name: MyINPulse-keycloak |     container_name: MyINPulse-keycloak | ||||||
|   | |||||||
| @@ -1,22 +0,0 @@ | |||||||
| POSTGRES_DB=postgres_db |  | ||||||
| POSTGRES_USER=postgres |  | ||||||
| POSTGRES_PASSWORD=postgres_db_user_password |  | ||||||
|  |  | ||||||
| KEYCLOAK_ADMIN=admin |  | ||||||
| KEYCLOAK_ADMIN_PASSWORD=admin |  | ||||||
| KEYCLOAK_HOSTNAME=localhost |  | ||||||
| KEYCLOAK_DB=keycloak_db |  | ||||||
| KEYCLOAK_USER=keycloak_db_user |  | ||||||
| KEYCLOAK_PASSWORD=keycloak_db_user_password |  | ||||||
|  |  | ||||||
| BACKEND_DB=backend_db |  | ||||||
| BACKEND_USER=backend_db_user |  | ||||||
| BACKEND_PASSWORD=backend_db_user_password |  | ||||||
|  |  | ||||||
| DATABASE_URL=MyINPulse-DB |  | ||||||
|  |  | ||||||
| VITE_KEYCLOAK_URL=http://localhost:7080 |  | ||||||
| VITE_KEYCLOAK_CLIENT_ID=myinpulse-dev |  | ||||||
| VITE_KEYCLOAK_REALM=test |  | ||||||
| VITE_APP_URL=http://localhost:5173 |  | ||||||
| VITE_BACKEND_URL=http://localhost:8081/ |  | ||||||
							
								
								
									
										5
									
								
								config/frontdev.front.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								config/frontdev.front.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | VITE_KEYCLOAK_URL=http://localhost:7080 | ||||||
|  | VITE_KEYCLOAK_CLIENT_ID=myinpulse-dev | ||||||
|  | VITE_KEYCLOAK_REALM=test | ||||||
|  | VITE_APP_URL=http://localhost:5173 | ||||||
|  | VITE_BACKEND_URL=http://localhost:8081/ | ||||||
							
								
								
									
										6
									
								
								config/frontdev.main.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								config/frontdev.main.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | POSTGRES_DB=keycloak_db | ||||||
|  | POSTGRES_USER=keycloak_db_user | ||||||
|  | POSTGRES_PASSWORD=keycloak_db_user_password | ||||||
|  | KEYCLOAK_ADMIN=admin | ||||||
|  | KEYCLOAK_ADMIN_PASSWORD=admin | ||||||
|  | KEYCLOAK_HOSTNAME=localhost | ||||||
| @@ -1,14 +1,11 @@ | |||||||
| services: | services: | ||||||
|   postgres: |   postgres: | ||||||
|     env_file: .env |     image: postgres:latest | ||||||
|     build: |  | ||||||
|       context: postgres/ |  | ||||||
|       dockerfile: Dockerfile |  | ||||||
|     container_name: MyINPulse-DB |     container_name: MyINPulse-DB | ||||||
|     #ports: |     #ports: | ||||||
|     #  - 5432:5432 |     #  - 5432:5432 | ||||||
|     volumes: |     volumes: | ||||||
|       - ./postgres/data:/var/lib/postgresql/data |       - ./postgres:/var/lib/postgresql/data | ||||||
|     environment: |     environment: | ||||||
|       POSTGRES_DB: ${POSTGRES_DB} |       POSTGRES_DB: ${POSTGRES_DB} | ||||||
|       POSTGRES_USER: ${POSTGRES_USER} |       POSTGRES_USER: ${POSTGRES_USER} | ||||||
| @@ -30,10 +27,10 @@ services: | |||||||
|       KC_BOOTSTRAP_ADMIN_USERNAME: ${KEYCLOAK_ADMIN} |       KC_BOOTSTRAP_ADMIN_USERNAME: ${KEYCLOAK_ADMIN} | ||||||
|       KC_BOOTSTRAP_ADMIN_PASSWORD: ${KEYCLOAK_ADMIN_PASSWORD} |       KC_BOOTSTRAP_ADMIN_PASSWORD: ${KEYCLOAK_ADMIN_PASSWORD} | ||||||
|       KC_LOG_LEVEL: info |       KC_LOG_LEVEL: info | ||||||
|     command: ["start-dev", "--http-port", "7080", "--https-port", "7443", "--hostname", "${KEYCLOAK_HOSTNAME}"] # TODO: remove start-dev |     command: ["start-dev", "--http-port", "7080", "--https-port", "7443", "--hostname", "${KEYCLOAK_HOSTNAME}"] | ||||||
|     #ports: |     ports: | ||||||
|     #  - "7080:7080" |       - "7080:7080" | ||||||
|     #  - "7443:7443" |       - "7443:7443" | ||||||
|     depends_on: |     depends_on: | ||||||
|       - postgres |       - postgres | ||||||
|  |  | ||||||
| @@ -50,6 +47,6 @@ services: | |||||||
|       context: ./MyINPulse-back/ |       context: ./MyINPulse-back/ | ||||||
|       dockerfile: Dockerfile |       dockerfile: Dockerfile | ||||||
|     container_name: MyINPulse-back |     container_name: MyINPulse-back | ||||||
|     #ports: |     ports: | ||||||
|     #  - "8081:8080" |       - "8081:8080" | ||||||
|    |    | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| POSTGRES_DB=postgres_db |  | ||||||
| POSTGRES_USER=postgres |  | ||||||
| POSTGRES_PASSWORD=postgres_db_user_password |  | ||||||
|  |  | ||||||
| KEYCLOAK_ADMIN=admin |  | ||||||
| KEYCLOAK_ADMIN_PASSWORD=admin |  | ||||||
| KEYCLOAK_HOSTNAME=0549cd63f912d5dc9b31278d6f.eirb.fr |  | ||||||
| KEYCLOAK_DB=keycloak_db |  | ||||||
| KEYCLOAK_USER=keycloak_db_user |  | ||||||
| KEYCLOAK_PASSWORD=keycloak_db_user_password |  | ||||||
|  |  | ||||||
| BACKEND_DB=backend_db |  | ||||||
| BACKEND_USER=backend_db_user |  | ||||||
| BACKEND_PASSWORD=backend_db_user_password |  | ||||||
|  |  | ||||||
| DATABASE_URL=MyINPulse-DB |  | ||||||
|  |  | ||||||
| VITE_KEYCLOAK_URL=https://0549cd63f912d5dc9b31278d6f.eirb.fr |  | ||||||
| VITE_KEYCLOAK_CLIENT_ID=myinpulse-eirb |  | ||||||
| VITE_KEYCLOAK_REALM=test |  | ||||||
| VITE_APP_URL=https://0549cd63f912d5dc9b31278d6f.piair.dev |  | ||||||
| VITE_BACKEND_URL=http://TODO/ |  | ||||||
							
								
								
									
										5
									
								
								config/prod.front.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								config/prod.front.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | VITE_KEYCLOAK_URL=https://0549cd63f912d5dc9b31278d6f.eirb.fr | ||||||
|  | VITE_KEYCLOAK_CLIENT_ID=myinpulse-eirb | ||||||
|  | VITE_KEYCLOAK_REALM=test | ||||||
|  | VITE_APP_URL=https://0549cd63f912d5dc9b31278d6f.piair.dev | ||||||
|  | VITE_BACKEND_URL=http://TODO/ | ||||||
							
								
								
									
										6
									
								
								config/prod.main.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								config/prod.main.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | POSTGRES_DB=keycloak_db | ||||||
|  | POSTGRES_USER=keycloak_db_user | ||||||
|  | POSTGRES_PASSWORD=keycloak_db_user_password | ||||||
|  | KEYCLOAK_ADMIN=admin | ||||||
|  | KEYCLOAK_ADMIN_PASSWORD=admin | ||||||
|  | KEYCLOAK_HOSTNAME=0549cd63f912d5dc9b31278d6f.eirb.fr | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| Format des comptes rendus de réunion : |  | ||||||
| Texte organisé par bullet point, chaque bullet point est séparé par "//" pour pouvoir être correctement généré. |  | ||||||
|  |  | ||||||
| Exemple : |  | ||||||
| Le texte "// blablabla // oui bonjour" |  | ||||||
| donne le résultat |  | ||||||
|  |  | ||||||
| Point n°1 : |  | ||||||
|   blablabla |  | ||||||
|  |  | ||||||
| Point n°2 : |  | ||||||
|   oui bonjour |  | ||||||
							
								
								
									
										0
									
								
								front/Dockerfile
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										0
									
								
								front/Dockerfile
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
								
								
									
										2
									
								
								front/MyINPulse-front/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								front/MyINPulse-front/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -36,4 +36,4 @@ playwright-report/ | |||||||
| # Custom | # Custom | ||||||
|  |  | ||||||
| .installed | .installed | ||||||
| ./package-lock.json | package-lock.json | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| { |  | ||||||
|   "useTabs":false, |  | ||||||
|   "semi":true, |  | ||||||
|   "trailingComma":"es5", |  | ||||||
|   "arrowParens":"always", |  | ||||||
|   "tabWidth":4 |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| import eslint from "@eslint/js"; |  | ||||||
| import eslintConfigPrettier from "eslint-config-prettier"; |  | ||||||
| import eslintPluginVue from "eslint-plugin-vue"; |  | ||||||
| import globals from "globals"; |  | ||||||
| import typescriptEslint from "typescript-eslint"; |  | ||||||
|  |  | ||||||
| export default typescriptEslint.config( |  | ||||||
|     { ignores: ["*.d.ts", "**/coverage", "**/dist"] }, |  | ||||||
|     { |  | ||||||
|         extends: [ |  | ||||||
|             eslint.configs.recommended, |  | ||||||
|             ...typescriptEslint.configs.recommended, |  | ||||||
|             ...eslintPluginVue.configs["flat/recommended"], |  | ||||||
|         ], |  | ||||||
|         files: ["**/*.{ts,vue}"], |  | ||||||
|         languageOptions: { |  | ||||||
|             ecmaVersion: "latest", |  | ||||||
|             sourceType: "module", |  | ||||||
|             globals: globals.browser, |  | ||||||
|             parserOptions: { |  | ||||||
|                 parser: typescriptEslint.parser, |  | ||||||
|             }, |  | ||||||
|         }, |  | ||||||
|         rules: { |  | ||||||
|             // your rules |  | ||||||
|         }, |  | ||||||
|     }, |  | ||||||
|     eslintConfigPrettier |  | ||||||
| ); |  | ||||||
| @@ -1,63 +1,10 @@ | |||||||
| { | { | ||||||
|   "entrepreneurs": [ |   "entrepreneurs": [ | ||||||
|     { "id": 1, "name": "Alice", "email": "alice@example.com" }, |       { "id": 1, "name": "Alice", "email": "alice@example.com" }, | ||||||
|     { "id": 2, "name": "Bob", "email": "bob@example.com" }, |       { "id": 2, "name": "Bob", "email": "bob@example.com" }, | ||||||
|     { "id": 3, "name": "Charlie", "email": "charlie@example.com" } |       { "id": 3, "name": "Charlie", "email": "charlie@example.com" } | ||||||
|   ], |   ], | ||||||
|   "data": [ |   "data": [ | ||||||
|     { |       { "canva_data": "this is a fake data to test api" } | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 1, |  | ||||||
|       "title_text": "1. Problème", |  | ||||||
|       "description": "3 problèmes essentiels à résoudre pour le client" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 2, |  | ||||||
|       "title_text": "2. Segments", |  | ||||||
|       "description": "Les segments de clientèle visés" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 3, |  | ||||||
|       "title_text": "3. Valeur", |  | ||||||
|       "description": "La proposition de valeur" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 4, |  | ||||||
|       "title_text": "4. Solution", |  | ||||||
|       "description": "Les solutions proposées" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 5, |  | ||||||
|       "title_text": "5. Avantage", |  | ||||||
|       "description": "Les avantages concurrentiels" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 6, |  | ||||||
|       "title_text": "6. Canaux", |  | ||||||
|       "description": "Les canaux de distribution" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 7, |  | ||||||
|       "title_text": "7. Indicateurs", |  | ||||||
|       "description": "Les indicateurs clés de performance" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 8, |  | ||||||
|       "title_text": "8. Coûts", |  | ||||||
|       "description": "Les coûts associés" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "projectId": 1, |  | ||||||
|       "title": 9, |  | ||||||
|       "title_text": "9. Revenus", |  | ||||||
|       "description": "Les sources de revenus" |  | ||||||
|     } |  | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1397
									
								
								front/MyINPulse-front/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1397
									
								
								front/MyINPulse-front/package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -26,15 +26,8 @@ | |||||||
|     "@types/node": "^22.10.7", |     "@types/node": "^22.10.7", | ||||||
|     "@vitejs/plugin-vue": "^5.2.1", |     "@vitejs/plugin-vue": "^5.2.1", | ||||||
|     "@vue/tsconfig": "^0.7.0", |     "@vue/tsconfig": "^0.7.0", | ||||||
|     "eslint": "^9.20.0", |  | ||||||
|     "eslint-config-prettier": "^10.0.1", |  | ||||||
|     "eslint-plugin-vue": "^9.32.0", |  | ||||||
|     "globals": "^15.14.0", |  | ||||||
|     "jiti": "^2.4.2", |  | ||||||
|     "npm-run-all2": "^7.0.2", |     "npm-run-all2": "^7.0.2", | ||||||
|     "prettier": "3.5.0", |  | ||||||
|     "typescript": "~5.7.3", |     "typescript": "~5.7.3", | ||||||
|     "typescript-eslint": "^8.23.0", |  | ||||||
|     "vite": "^6.0.11", |     "vite": "^6.0.11", | ||||||
|     "vite-plugin-vue-devtools": "^7.7.0", |     "vite-plugin-vue-devtools": "^7.7.0", | ||||||
|     "vue-tsc": "^2.2.0" |     "vue-tsc": "^2.2.0" | ||||||
|   | |||||||
| @@ -1,26 +1,26 @@ | |||||||
| <template> | <template> | ||||||
|     <header> |     <header> | ||||||
|         <img src="./icons/logo inpulse.png" alt="INPulse" /> |       <img src="./icons/logo inpulse.png" alt="INPulse" /> | ||||||
|     </header> |     </header> | ||||||
| </template> |   </template> | ||||||
| 
 |    | ||||||
| <script lang="ts"> |   <script lang="ts"> | ||||||
| export default { |   export default { | ||||||
|     name: "HeaderComponent", |     name: 'Header', | ||||||
| }; |   }; | ||||||
| </script> |   </script> | ||||||
| 
 |    | ||||||
| <style scoped> |   <style scoped> | ||||||
| header img { |   header img { | ||||||
|     width: 100px; |     width: 100px; | ||||||
| } |   } | ||||||
| 
 | 
 | ||||||
| header { |   header{ | ||||||
|     display: flex; |     display: flex; | ||||||
|     justify-content: space-between; |     justify-content: space-between; | ||||||
|     align-items: center; |     align-items: center; | ||||||
|     padding: 10px 20px; |     padding: 10px 20px; | ||||||
|     background-color: #fff; |     background-color: #fff; | ||||||
|     border-bottom: 2px solid #ddd; |     border-bottom: 2px solid #ddd; | ||||||
| } |   } | ||||||
| </style> |   </style> | ||||||
| @@ -15,7 +15,6 @@ | |||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import { defineProps } from "vue"; | import { defineProps } from "vue"; | ||||||
| import { useRouter } from 'vue-router' | import { useRouter } from 'vue-router' | ||||||
							
								
								
									
										326
									
								
								front/MyINPulse-front/src/components/canvas/CanvasItem.vue
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										326
									
								
								front/MyINPulse-front/src/components/canvas/CanvasItem.vue
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							| @@ -1,330 +1,88 @@ | |||||||
| <template> | <template> | ||||||
|   <div :class="['cell', { expanded }]" @click="handleClick"> |     <div :class="['cell', { expanded }]" | ||||||
|     <h3>{{ title_text }}</h3> |     @click="toggleExpand" | ||||||
|  |     :style="{ justifyContent: expanded ? 'flex-start' : 'center' }"> <!-- Looking for finding a way  | ||||||
|  |                                                           to make this style in the toggleExpand event --> | ||||||
|  |  | ||||||
|     <div class="section-bloc" v-for="(desc, index) in currentDescriptions" :key="index"> |       <h3>{{ title }}</h3> | ||||||
|       <!-- Mode affichage --> |       <p>{{ currentDescription }}</p> | ||||||
|       <template v-if="!isEditing[index]"> |  | ||||||
|         <div class="description"> |  | ||||||
|           <p>{{ desc }}</p> |  | ||||||
|         </div> |  | ||||||
|         <div class="button-container"> |  | ||||||
|           <button v-if="expanded" @click.stop="startEditing(index)" class="edit-button">Éditer</button> |  | ||||||
|         </div> |  | ||||||
|       </template> |  | ||||||
|  |  | ||||||
|       <!-- Mode édition --> |  | ||||||
|       <template v-else> |  | ||||||
|         <textarea v-model="editedDescriptions[index]" class="edit-input"></textarea> |  | ||||||
|         <div class="button-container"> |  | ||||||
|           <button @click.stop="saveEdit(index)" class="save-button">Enregistrer</button> |  | ||||||
|           <button @click.stop="cancelEdit(index)" class="cancel-button">Annuler</button> |  | ||||||
|         </div> |  | ||||||
|       </template> |  | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|   </div> |  | ||||||
| </template> | </template> | ||||||
|  |    | ||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import { ref, defineProps } from "vue"; | import { ref, defineProps, onMounted } from "vue"; | ||||||
| import axios from "axios"; | import axios from "axios"; | ||||||
|  |  | ||||||
| const IS_MOCK_MODE = true;  | const props = defineProps<{   | ||||||
|  |     title: string; | ||||||
| const props = defineProps<{ |     description: string; | ||||||
|   projectId: number; |  | ||||||
|   title: number; |  | ||||||
|   title_text: string; |  | ||||||
|   description: string; |  | ||||||
| }>(); | }>(); | ||||||
|  |  | ||||||
| const expanded = ref(false); | const expanded = ref(false); | ||||||
| const currentDescriptions = ref<string[]>([]); | const currentDescription = ref(props.description); | ||||||
| currentDescriptions.value[0] = props.description; |  | ||||||
| const editedDescriptions = ref<string[]>([]); |  | ||||||
| const isEditing = ref<boolean[]>([]); |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /* |  | ||||||
| const fetchData = async () => { | const fetchData = async () => { | ||||||
|   try { |   try { | ||||||
|     const response = await axios.get("http://localhost:5000/data"); // Met à jour l'URL |     const response = await axios.get("http://localhost:5000/data"); // Update the URL if needed | ||||||
|     if (response.data.length > 0) { |     currentDescription.value = response.data[0].canva_data; | ||||||
|       currentDescription.value = response.data[0].canva_data; |  | ||||||
|       editedDescription.value = response.data[0].canva_data; |  | ||||||
|     } else { |  | ||||||
|       console.warn("Aucune donnée reçue."); |  | ||||||
|     } |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error("Erreur lors de la récupération des données :", error); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| // Fonction de simulation de l'API |  | ||||||
| const mockFetch = async (projectId: number, title: number, date: string) => { |  | ||||||
|   console.log(`Mock fetch pour projectId: ${projectId}, title: ${title}, date: ${date}`); |  | ||||||
|  |  | ||||||
|   return new Promise<{ txt: string }[]>((resolve) => { |  | ||||||
|     setTimeout(() => { |  | ||||||
|       resolve([ |  | ||||||
|         { txt: "Ceci est une description 1 pour tester the damn front, Why are u still reading dumbass this is just some nonsense sentence XD" }, |  | ||||||
|         { txt: "Ceci est une description 1 pour tester the damn front, Bruh are u still here?" }, |  | ||||||
|         { txt: "Ceci est une description 1 pour tester the damn front, .-. BRUH" } |  | ||||||
|       ]); |  | ||||||
|     }, 500); // Simule un délai réseau de 500ms |  | ||||||
|   }); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Fonction fetchData avec possibilité d'utiliser le mock |  | ||||||
| const fetchData = async (projectId: number, title: number, date: string, useMock = false) => { |  | ||||||
|   try { |  | ||||||
|     const responseData = useMock |  | ||||||
|       ? await mockFetch(projectId, title, date) |  | ||||||
|       : (await axios.get<{ txt: string }[]>( |  | ||||||
|           `http://localhost:5000/shared/projects/lcsection/${projectId}/${title}/${date}` |  | ||||||
|         )).data; |  | ||||||
|  |  | ||||||
|     if (responseData.length > 0) { |  | ||||||
|       currentDescriptions.value = responseData.map((item) => item.txt); |  | ||||||
|       editedDescriptions.value = [...currentDescriptions.value]; |  | ||||||
|       isEditing.value = Array(responseData.length).fill(false); |  | ||||||
|     } else { |  | ||||||
|       console.warn("Aucune donnée reçue."); |  | ||||||
|     } |  | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     console.error("Erreur lors de la récupération des données :", error); |     console.error("Erreur lors de la récupération des données :", error); | ||||||
|   } |   } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| // Utilisation du mock dans handleClick pour tester sans serveur | const toggleExpand = async () => { | ||||||
| const handleClick = async () => { |  | ||||||
|   if (!expanded.value) { |   if (!expanded.value) { | ||||||
|     await fetchData(props.projectId, props.title, "NaN", IS_MOCK_MODE); // true pour activer le mock |     await fetchData(); | ||||||
|   } else if (!isEditing.value.includes(true)) { |  | ||||||
|     // Réinitialiser les descriptions si aucune édition n'est en cours |  | ||||||
|     currentDescriptions.value = [props.description]; |  | ||||||
|     editedDescriptions.value = [props.description]; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (!isEditing.value.includes(true)) { |  | ||||||
|     expanded.value = !expanded.value; |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| const startEditing = (index: number) => { |  | ||||||
|   isEditing.value[index] = true; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /* |  | ||||||
| const saveEdit = async (index: number) => { |  | ||||||
|   try { |  | ||||||
|     const id = index + 1; // À adapter selon l'ID réel des données |  | ||||||
|     await axios.put(`http://localhost:5000/data/${id}`, { |  | ||||||
|       canva_data: editedDescriptions.value[index] |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     // Mettre à jour l'affichage local après la mise à jour réussie |  | ||||||
|     currentDescriptions.value[index] = editedDescriptions.value[index]; |  | ||||||
|     isEditing.value[index] = false; |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error("Erreur lors de la mise à jour des données :", error); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| // Fonction de mock pour l'enregistrement |  | ||||||
| const mockSaveEdit = async (index: number) => { |  | ||||||
|   try { |  | ||||||
|     const id = index + 1; // À adapter selon l'ID réel des données |  | ||||||
|     console.log(`Mock save pour l'ID ${id} avec la description : ${editedDescriptions.value[index]}`); |  | ||||||
|  |  | ||||||
|     // Simuler un délai d'enregistrement comme une requête réseau |  | ||||||
|     await new Promise((resolve) => setTimeout(resolve, 500)); // Simulation de délai réseau |  | ||||||
|  |  | ||||||
|     // Mettre à jour l'affichage local après la mise à jour réussie |  | ||||||
|     currentDescriptions.value[index] = editedDescriptions.value[index]; |  | ||||||
|     isEditing.value[index] = false; |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error("Erreur lors de la mise à jour des données mockées :", error); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Utilisation de `mockSaveEdit` au lieu de `saveEdit` dans `handleClick` ou tout autre endroit |  | ||||||
| const saveEdit = async (index: number) => { |  | ||||||
|   if (IS_MOCK_MODE) { |  | ||||||
|     await mockSaveEdit(index); |  | ||||||
|   } else { |   } else { | ||||||
|     try { |     currentDescription.value = props.description; | ||||||
|       const id = index + 1; |  | ||||||
|       await axios.put(`http://localhost:5000/data/${id}`, { |  | ||||||
|         canva_data: editedDescriptions.value[index] |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       // Mettre à jour l'affichage local après la mise à jour réussie |  | ||||||
|       currentDescriptions.value[index] = editedDescriptions.value[index]; |  | ||||||
|       isEditing.value[index] = false; |  | ||||||
|     } catch (error) { |  | ||||||
|       console.error("Erreur lors de la mise à jour des données :", error); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |   expanded.value = !expanded.value; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const cancelEdit = (index: number) => { |  | ||||||
|   editedDescriptions.value[index] = currentDescriptions.value[index]; |  | ||||||
|   isEditing.value[index] = false; |  | ||||||
| }; |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  |    | ||||||
| <style scoped> | <style scoped> | ||||||
| @import "@/components/canvas/style-project.css"; | @import "@/components/canvas/style-project.css"; | ||||||
|  |  | ||||||
| .cell { | .cell { | ||||||
|   display: flex; | 	display: flex; | ||||||
|   flex-direction: column; | 	flex-direction: column; | ||||||
|   align-items: center; | 	align-items: center; | ||||||
|   justify-content: center; | 	justify-content: center; | ||||||
|   text-align: center; | 	text-align: center; | ||||||
|   transition: all 0.3s ease; | 	transition: all 0.3s ease; | ||||||
|   cursor: pointer; | 	cursor: pointer; | ||||||
|   box-shadow: 0 4px 5px rgba(0, 0, 0, 0.1); |   box-shadow: 0 4px 5px rgba(0, 0, 0, 0.1); | ||||||
| } | } | ||||||
|  |  | ||||||
| .expanded-content { |  | ||||||
|   justify-content: flex-start !important; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .cell:not(.expanded):hover { | .cell:not(.expanded):hover { | ||||||
|   transform: scale(1.05); |   transform: scale(1.05); | ||||||
|   box-shadow: 0 8px 9px rgba(0, 0, 0, 0.2); |   box-shadow: 0 8px 9px rgba(0, 0, 0, 0.2); | ||||||
| } | } | ||||||
|  |  | ||||||
| .cell h3 { | .cell h3 { | ||||||
|   font-size: 20px; |     font-size: 20px; | ||||||
|   font-weight: 500; |     font-weight: 500; | ||||||
|   /*margin-bottom: 10px;*/ |     /*margin-bottom: 10px;*/ | ||||||
| } | } | ||||||
|  |  | ||||||
| .cell p { | .cell p { | ||||||
|   font-size: 14px; |     font-size: 14px; | ||||||
|   color: #666; |     color: #666; | ||||||
| } | } | ||||||
|  |  | ||||||
| .expanded { | .expanded { | ||||||
|   padding-top: 10%; |  | ||||||
|   position: fixed; |   position: fixed; | ||||||
|   top: 0; | 	top: 0; | ||||||
|   left: 0; | 	left: 0; | ||||||
|   width: 100%; | 	width: 100%; | ||||||
|   height: 100%; | 	height: 100%; | ||||||
|   background: white; | 	background: white; | ||||||
|   z-index: 10; | 	z-index: 10; | ||||||
|   display: flex; | 	display: flex; | ||||||
|   align-items: center; | 	align-items: center; | ||||||
|   justify-content: flex-start; | 	justify-content: center; | ||||||
|   box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); | 	box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); | ||||||
| } | } | ||||||
|  | </style> | ||||||
|  |  | ||||||
| .description { |  | ||||||
|   display: flex; |  | ||||||
|   align-items: center;  /* Centre verticalement */ |  | ||||||
|   justify-content: center; /* Centre horizontalement */ |  | ||||||
|   text-align: center; /* Centre le texte à l'intérieur */ |  | ||||||
|   width: 100%; |  | ||||||
|   height: 100%; |  | ||||||
|   font-size: 16px; |  | ||||||
|   margin-top: 10px; |  | ||||||
|   margin-left: 2%; |  | ||||||
|   margin-right: 4%; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .description + .p { |  | ||||||
|   align-items: center; |  | ||||||
|   justify-content: center; |  | ||||||
|   text-align: center; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .edit-input { |  | ||||||
|   width: 100%; |  | ||||||
|   height: 100%; |  | ||||||
|   font-size: 16px; |  | ||||||
|   padding: 10px; |  | ||||||
|   border: 1px solid #ccc; |  | ||||||
|   border-radius: 5px; |  | ||||||
|   margin-top: 10px; |  | ||||||
|   box-sizing: border-box; |  | ||||||
|   margin-left: 2%; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| .button-container { |  | ||||||
|   display: block; |  | ||||||
|   margin-top: 20px; |  | ||||||
|   justify-content: center; |  | ||||||
|   align-items: center; |  | ||||||
|   gap: 10px; |  | ||||||
|   padding-right: 1%; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .section-bloc ,.editing-section-bloc { |  | ||||||
|   width: 100%; |  | ||||||
|   justify-content: center; |  | ||||||
|   align-items: center; |  | ||||||
|   display: flex; |  | ||||||
|   margin-right: 10%; |  | ||||||
|   margin: 10px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .edit-button { |  | ||||||
|   width: 100px; |  | ||||||
|   height: 40px; |  | ||||||
|   border: none; |  | ||||||
|   border-radius: 5px; |  | ||||||
|   cursor: pointer; |  | ||||||
|   transition: background 0.3s ease; |  | ||||||
|   font-size: 12px; |  | ||||||
|   margin-right: 20px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .save-button, .cancel-button { |  | ||||||
|   width: 100px; |  | ||||||
|   height: 40px; |  | ||||||
|   border: none; |  | ||||||
|   border-radius: 5px; |  | ||||||
|   cursor: pointer; |  | ||||||
|   transition: background 0.3s ease; |  | ||||||
|   font-size: 12px; |  | ||||||
|   margin-bottom: 5px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .edit-button { |  | ||||||
|   background-color: #007bff; |  | ||||||
|   color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .save-button { |  | ||||||
|   background-color: #28a745; |  | ||||||
|   color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .cancel-button { |  | ||||||
|   background-color: #dc3545; |  | ||||||
|   color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .edit-button:hover { |  | ||||||
|   background-color: #0056b3; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .save-button:hover { |  | ||||||
|   background-color: #218838; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .cancel-button:hover { |  | ||||||
|   background-color: #c82333; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| </style> |  | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ | |||||||
| </template> | </template> | ||||||
|  |  | ||||||
|  |  | ||||||
| <script setup lang="ts"> | <script setup> | ||||||
| import { ref, onMounted } from "vue"; | import { ref, onMounted } from "vue"; | ||||||
| import axios from "axios"; | import axios from "axios"; | ||||||
|  |  | ||||||
| @@ -34,7 +34,7 @@ const toggleDropdown = () => { | |||||||
| const fetchEntrepreneurs = async () => { | const fetchEntrepreneurs = async () => { | ||||||
|   try { |   try { | ||||||
|     const response = await axios.get("http://localhost:5000/entrepreneurs"); |     const response = await axios.get("http://localhost:5000/entrepreneurs"); | ||||||
|     entrepreneurEmails.value = response.data.map((e: { email: string }) => e.email); |     entrepreneurEmails.value = response.data.map(e => e.email); | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     console.error("Erreur lors de la récupération des entrepreneurs:", error); |     console.error("Erreur lors de la récupération des entrepreneurs:", error); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -4,9 +4,7 @@ | |||||||
|       v-for="(item, index) in items" |       v-for="(item, index) in items" | ||||||
|       :key="index" |       :key="index" | ||||||
|       :title="item.title" |       :title="item.title" | ||||||
|       :title_text="item.title_text" |  | ||||||
|       :description="item.description" |       :description="item.description" | ||||||
|       :projectId="item.projectId" |  | ||||||
|       :class="item.class" |       :class="item.class" | ||||||
|     /> |     /> | ||||||
|   </div> |   </div> | ||||||
| @@ -17,17 +15,16 @@ import { ref } from "vue"; | |||||||
| import CanvasItem from "@/components/canvas/CanvasItem.vue"; | import CanvasItem from "@/components/canvas/CanvasItem.vue"; | ||||||
|  |  | ||||||
| const items = ref([ | const items = ref([ | ||||||
|   { projectId: 1, title: 1, title_text: "1. Problème", description: "3 problèmes essentiels à résoudre pour le client", class: "Probleme" }, |   { title: "1. Problème", description: "3 problèmes essentiels à résoudre pour le client", class: "Probleme" }, | ||||||
|   { projectId: 1, title: 2, title_text: "2. Segments", description: "Les segments de clientèle visés", class: "Segments" }, |   { title: "2. Segments", description: "Les segments de clientèle visés", class: "Segments" }, | ||||||
|   { projectId: 1, title: 3, title_text: "3. Valeur", description: "La proposition de valeur", class: "Valeur" }, |   { title: "3. Valeur", description: "La proposition de valeur", class: "Valeur" }, | ||||||
|   { projectId: 1, title: 4, title_text: "4. Solution", description: "Les solutions proposées", class: "Solution" }, |   { title: "4. Solution", description: "Les solutions proposées", class: "Solution" }, | ||||||
|   { projectId: 1, title: 5, title_text: "5. Avantage", description: "Les avantages concurrentiels", class: "Avantage" }, |   { title: "5. Avantage", description: "Les avantages concurrentiels", class: "Avantage" }, | ||||||
|   { projectId: 1, title: 6, title_text: "6. Canaux", description: "Les canaux de distribution", class: "Canaux" }, |   { title: "6. Canaux", description: "Les canaux de distribution", class: "Canaux" }, | ||||||
|   { projectId: 1, title: 7, title_text: "7. Indicateurs", description: "Les indicateurs clés de performance", class: "Indicateurs" }, |   { title: "7. Indicateurs", description: "Les indicateurs clés de performance", class: "Indicateurs" }, | ||||||
|   { projectId: 1, title: 8, title_text: "8. Coûts", description: "Les coûts associés", class: "Couts" }, |   { title: "8. Coûts", description: "Les coûts associés", class: "Couts" }, | ||||||
|   { projectId: 1, title: 9, title_text: "9. Revenus", description: "Les sources de revenus", class: "Revenus" } |   { title: "9. Revenus", description: "Les sources de revenus", class: "Revenus" } | ||||||
| ]); | ]); | ||||||
|  |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <style scoped> | <style scoped> | ||||||
| @@ -47,13 +44,13 @@ const items = ref([ | |||||||
|   overflow: auto; |   overflow: auto; | ||||||
| } | } | ||||||
|  |  | ||||||
| .Probleme { grid-column: 1 / 3; grid-row: 1 / 5; padding-top: 20px;} | .Probleme { grid-column: 1 / 3; grid-row: 1 / 5; } | ||||||
| .Segments { grid-column: 9 / 11; grid-row: 1 / 5; padding-top: 20px;} | .Segments { grid-column: 9 / 11; grid-row: 1 / 5; } | ||||||
| .Valeur { grid-column: 5 / 7; grid-row: 1 / 5; padding-top: 20px;} | .Valeur { grid-column: 5 / 7; grid-row: 1 / 5; } | ||||||
| .Solution { grid-column: 3 / 5; grid-row: 1 / 3; padding-top: 20px;} | .Solution { grid-column: 3 / 5; grid-row: 1 / 3; } | ||||||
| .Avantage { grid-column: 7 / 9; grid-row: 1 / 3; padding-top: 20px;} | .Avantage { grid-column: 7 / 9; grid-row: 1 / 3; } | ||||||
| .Canaux { grid-column: 7 / 9; grid-row: 3 / 5; padding-top: 20px;} | .Canaux { grid-column: 7 / 9; grid-row: 3 / 5; } | ||||||
| .Indicateurs { grid-column: 3 / 5; grid-row: 3 / 5; padding-top: 20px;} | .Indicateurs { grid-column: 3 / 5; grid-row: 3 / 5; } | ||||||
| .Couts { grid-column: 1 / 6; grid-row: 5 / 7; padding-top: 20px;} | .Couts { grid-column: 1 / 6; grid-row: 5 / 7; } | ||||||
| .Revenus { grid-column: 6 / 11; grid-row: 5 / 7; padding-top: 20px;} | .Revenus { grid-column: 6 / 11; grid-row: 5 / 7; } | ||||||
| </style> | </style> | ||||||
|   | |||||||
| @@ -1,43 +1,17 @@ | |||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import { color } from "@/services/popupDisplayer.ts"; | const props = defineProps(['data']); | ||||||
| import type { PropType } from "vue"; |  | ||||||
|  |  | ||||||
| defineProps({ |  | ||||||
|     errorMessage: { |  | ||||||
|         type: Object as PropType<string>, |  | ||||||
|         required: true, |  | ||||||
|     }, |  | ||||||
|     errorColor: { |  | ||||||
|         type: Object as PropType<color>, |  | ||||||
|         default: color.Red, |  | ||||||
|     }, |  | ||||||
| }); |  | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <template> | <template> | ||||||
|     <div |   <div :class='["red", "yellow", "blue", "green"][data.type]' class="error-modal"> | ||||||
|         :class="['red', 'yellow', 'blue', 'green'][errorColor]" |     <p>{{["Erreur :(", "Warning :|", "Info :)", "Succes ;)"][data.type]}}</p> | ||||||
|         class="error-modal" |     <p>{{data.errorMessage}}</p> | ||||||
|     > |     <div class="loading" :class='["red-loader", "yellow-loader", "blue-loader", "green-loader"][data.type]'></div> | ||||||
|         <p> |   </div> | ||||||
|             {{ |  | ||||||
|                 ["Erreur :(", "Warning :|", "Info :)", "Succes ;)"][errorColor] |  | ||||||
|             }} |  | ||||||
|         </p> |  | ||||||
|         <p>{{ errorMessage }}</p> |  | ||||||
|         <div |  | ||||||
|             class="loading" |  | ||||||
|             :class=" |  | ||||||
|                 ['red-loader', 'yellow-loader', 'blue-loader', 'green-loader'][ |  | ||||||
|                     errorColor |  | ||||||
|                 ] |  | ||||||
|             " |  | ||||||
|         ></div> |  | ||||||
|     </div> |  | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <style scoped> | <style scoped> | ||||||
| .error-modal { |   .error-modal{ | ||||||
|     margin-bottom: 1em; |     margin-bottom: 1em; | ||||||
|     padding: 1em; |     padding: 1em; | ||||||
|     border-radius: 1em; |     border-radius: 1em; | ||||||
| @@ -45,45 +19,41 @@ defineProps({ | |||||||
|     overflow: hidden; |     overflow: hidden; | ||||||
|     position: relative; |     position: relative; | ||||||
|     animation: disappear 5s linear forwards; |     animation: disappear 5s linear forwards; | ||||||
| } |   } | ||||||
|  |  | ||||||
| .red { |   .red{ | ||||||
|     background-color: #ee6055; |     background-color: #ee6055; | ||||||
|     color: white; |     color: white; | ||||||
| } |   } | ||||||
|  |   .red-loader { | ||||||
| .red-loader { |  | ||||||
|     background-color: #fa8383; |     background-color: #fa8383; | ||||||
| } |   } | ||||||
|  |  | ||||||
| .yellow { |   .yellow{ | ||||||
|     background-color: #ff9d23; |     background-color: #FF9D23; | ||||||
|     color: white; |      color: white; | ||||||
| } |    } | ||||||
|  |   .yellow-loader{ | ||||||
| .yellow-loader { |  | ||||||
|     background-color: #ffbf81; |     background-color: #ffbf81; | ||||||
| } |   } | ||||||
|  |  | ||||||
| .blue { |   .blue { | ||||||
|     background-color: #809bce; |     background-color: #809bce; | ||||||
|     color: white; |     color: white; | ||||||
| } |   } | ||||||
|  |   .blue-loader{ | ||||||
| .blue-loader { |  | ||||||
|     background-color: #95b8d1; |     background-color: #95b8d1; | ||||||
| } |   } | ||||||
|  |  | ||||||
| .green { |   .green { | ||||||
|     background-color: green; |     background-color: green; | ||||||
|     color: white; |     color: white; | ||||||
| } |   } | ||||||
|  |   .green-loader { | ||||||
| .green-loader { |  | ||||||
|     background-color: darkgreen; |     background-color: darkgreen; | ||||||
| } |   } | ||||||
|  |  | ||||||
| .loading { |   .loading { | ||||||
|     box-sizing: border-box; |     box-sizing: border-box; | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     padding: 0; |     padding: 0; | ||||||
| @@ -92,33 +62,33 @@ defineProps({ | |||||||
|     height: 1em; |     height: 1em; | ||||||
|     width: 0; |     width: 0; | ||||||
|     animation: loading 4s linear forwards; |     animation: loading 4s linear forwards; | ||||||
| } |   } | ||||||
|  |  | ||||||
| /* Animation for the loading bar */ |   /* Animation for the loading bar */ | ||||||
| @keyframes loading { |   @keyframes loading { | ||||||
|     0% { |     0% { | ||||||
|         width: 100%; |       width: 100%; | ||||||
|     } |     } | ||||||
|     100% { |     100% { | ||||||
|         width: 0; |       width: 0; | ||||||
|     } |     } | ||||||
| } |   } | ||||||
|  |  | ||||||
| @keyframes disappear { |   @keyframes disappear { | ||||||
|     0% { |     0% { | ||||||
|         height: 100px; |       height: 100px; | ||||||
|         padding: 1em; |       padding: 1em; | ||||||
|         margin: 1em; |       margin: 1em; | ||||||
|     } |     } | ||||||
|     80% { |     80% { | ||||||
|         height: 100px; |       height: 100px; | ||||||
|         padding: 1em; |       padding: 1em; | ||||||
|         margin: 1em; |       margin: 1em; | ||||||
|     } |     } | ||||||
|     100% { |     100% { | ||||||
|         height: 0; |       height: 0; | ||||||
|         margin: 0; |       margin: 0; | ||||||
|         padding: 0; |       padding: 0; | ||||||
|     } |     } | ||||||
| } |   } | ||||||
| </style> | </style> | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| <script setup lang="ts"> |  | ||||||
| import { addNewMessage } from "@/services/popupDisplayer.ts"; |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <template> |  | ||||||
|     <button |  | ||||||
|         @click=" |  | ||||||
|             addNewMessage( |  | ||||||
|                 'new error from another view', |  | ||||||
|                 Math.floor(Math.random() * 4) |  | ||||||
|             ) |  | ||||||
|         " |  | ||||||
|     > |  | ||||||
|         Add an error |  | ||||||
|     </button> |  | ||||||
| </template> |  | ||||||
|  |  | ||||||
| <style scoped></style> |  | ||||||
| @@ -1,30 +1,30 @@ | |||||||
| import { createApp } from "vue"; | import { createApp } from 'vue' | ||||||
| import App from "./App.vue"; | import App from './App.vue' | ||||||
| import router from "./router/router.ts"; | import router from './router/router.ts' | ||||||
| import { createPinia } from "pinia"; | import {createPinia} from "pinia"; | ||||||
| import piniaPluginPersistedstate from "pinia-plugin-persistedstate"; | import piniaPluginPersistedstate from 'pinia-plugin-persistedstate'; | ||||||
| import keycloakService from "./services/keycloak"; | import AuthStorePlugin from './plugins/authStore'; | ||||||
| import { type AuthStore, useAuthStore } from "@/stores/authStore.ts"; | import keycloakService from './services/keycloak'; | ||||||
|  | import {useAuthStore} from "@/stores/authStore.ts"; | ||||||
| let store: AuthStore; | let store: any; | ||||||
|  |  | ||||||
| keycloakService.CallInit(() => { | keycloakService.CallInit(() => { | ||||||
|     try { |     try { | ||||||
|         const app = createApp(App); |         const app = createApp(App) | ||||||
|  |  | ||||||
|         // Setup pinia store, allowing user to keep logged in status after refresh |         // Setup pinia store, allowing user to keep logged in status after refresh | ||||||
|         const pinia = createPinia(); |         const pinia = createPinia(); | ||||||
|         pinia.use(piniaPluginPersistedstate); |         pinia.use(piniaPluginPersistedstate); | ||||||
|         app.use(pinia); |         app.use(pinia); | ||||||
|  |         app.use(AuthStorePlugin, { pinia }); | ||||||
|         store = useAuthStore(); |         store = useAuthStore(); | ||||||
|         keycloakService.CallInitStore(store); |         app.use(router) | ||||||
|         app.use(router); |  | ||||||
|  |  | ||||||
|         app.mount("#app"); |         app.mount('#app'); | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
|         console.error("Error while initiating Keycloak."); |         console.error("Error while initiating Keycloak.") | ||||||
|         console.error(e); |         console.error(e) | ||||||
|         createApp(App).mount("#app"); |         createApp(App).mount('#app'); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| }) | }) | ||||||
| @@ -74,4 +74,4 @@ app.use(VueKeyCloak,{ | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| export { store }; | export {store}; | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { createRouter, createWebHistory } from "vue-router"; | import { createRouter, createWebHistory } from 'vue-router' | ||||||
|  |  | ||||||
| const router = createRouter({ | const router = createRouter({ | ||||||
|   history: createWebHistory(import.meta.env.BASE_URL), |   history: createWebHistory(import.meta.env.BASE_URL), | ||||||
| @@ -9,7 +9,7 @@ const router = createRouter({ | |||||||
|       // route level code-splitting |       // route level code-splitting | ||||||
|       // this generates a separate chunk (About.[hash].js) for this route |       // this generates a separate chunk (About.[hash].js) for this route | ||||||
|       // which is lazy-loaded when the route is visited. |       // which is lazy-loaded when the route is visited. | ||||||
|       component: () => import('../views/testComponent.vue'), |       component: () => import('../views/test.vue'), | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|     { |     { | ||||||
| @@ -27,4 +27,4 @@ const router = createRouter({ | |||||||
|   ], |   ], | ||||||
| }) | }) | ||||||
|  |  | ||||||
| export default router; | export default router | ||||||
|   | |||||||
| @@ -1,37 +1,31 @@ | |||||||
| import axios, { type AxiosError, type AxiosResponse } from "axios"; | import axios from "axios"; | ||||||
| import { store } from "@/main.ts"; | import {store} from "@/main.ts"; | ||||||
| import { addNewMessage, color } from "@/services/popupDisplayer.ts"; | import {addNewMessage, color} from "@/services/popupDisplayer.ts"; | ||||||
|  |  | ||||||
| const axiosInstance = axios.create({ | const axiosInstance = axios.create({ | ||||||
|     baseURL: import.meta.env.VITE_BACKEND_URL, |     baseURL: import.meta.env.VITE_BACKEND_URL, | ||||||
|     headers: { |     headers: { | ||||||
|         "Content-Type": "application/json", |         'Content-Type': 'application/json', | ||||||
|     }, |     }, | ||||||
| }); | }); | ||||||
|  |  | ||||||
| axiosInstance.interceptors.response.use( | axiosInstance.interceptors.response.use( | ||||||
|     (response) => response, // Directly return successful responses. |     response => response, // Directly return successful responses. | ||||||
|     async (error) => { |     async error => { | ||||||
|         const originalRequest = error.config; |         const originalRequest = error.config; | ||||||
|         if ( |         if (error.response.status === 401 && !originalRequest._retry && store.authenticated) { | ||||||
|             ((error.response && error.response.status === 401) || |  | ||||||
|                 error.code == "ERR_NETWORK") && |  | ||||||
|             !originalRequest._retry && |  | ||||||
|             store.authenticated |  | ||||||
|         ) { |  | ||||||
|             originalRequest._retry = true; // Mark the request as retried to avoid infinite loops. |             originalRequest._retry = true; // Mark the request as retried to avoid infinite loops. | ||||||
|             try { |             try { | ||||||
|                 await store.refreshUserToken(); |                 await store.refreshUserToken(); | ||||||
|                 // Update the authorization header with the new access token. |                 // Update the authorization header with the new access token. | ||||||
|                 axiosInstance.defaults.headers.common["Authorization"] = |                 axiosInstance.defaults.headers.common['Authorization'] = `Bearer ${store.user.token}`; | ||||||
|                     `Bearer ${store.user.token}`; |  | ||||||
|                 return axiosInstance(originalRequest); // Retry the original request with the new access token. |                 return axiosInstance(originalRequest); // Retry the original request with the new access token. | ||||||
|             } catch (refreshError) { |             } catch (refreshError) { | ||||||
|                 // Handle refresh token errors by clearing stored tokens and redirecting to the login page. |                 // Handle refresh token errors by clearing stored tokens and redirecting to the login page. | ||||||
|                 console.error("Token refresh failed:", refreshError); |                 console.error('Token refresh failed:', refreshError); | ||||||
|                 localStorage.removeItem("accessToken"); |                 localStorage.removeItem('accessToken'); | ||||||
|                 localStorage.removeItem("refreshToken"); |                 localStorage.removeItem('refreshToken'); | ||||||
|                 window.location.href = "/login"; |                 window.location.href = '/login'; | ||||||
|                 return Promise.reject(refreshError); |                 return Promise.reject(refreshError); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -40,29 +34,23 @@ axiosInstance.interceptors.response.use( | |||||||
| ); | ); | ||||||
|  |  | ||||||
| // TODO: spawn a error modal | // TODO: spawn a error modal | ||||||
| function defaultApiErrorHandler(err: AxiosError) { | function defaultApiErrorHandler(err: string){ | ||||||
|     addNewMessage(err.message, color.Red); |     addNewMessage(err, color.Red); | ||||||
| } | } | ||||||
|  |  | ||||||
| function defaultApiSuccessHandler(response: AxiosResponse) { | function defaultApiSuccessHandler(response: any){ | ||||||
|     addNewMessage(response.data, color.Green); |     addNewMessage(response.data, color.green) | ||||||
|  | } | ||||||
|  | function callApi(endpoint: string, onSuccessHandler?: any, onErrorHandler?: any): void { | ||||||
|  |     axiosInstance.get(endpoint).then( | ||||||
|  |         onSuccessHandler == null ? defaultApiSuccessHandler : onSuccessHandler | ||||||
|  |     ).catch( | ||||||
|  |         (err) => { | ||||||
|  |             onErrorHandler == null ? defaultApiErrorHandler(err): onErrorHandler(err); | ||||||
|  |             throw err; | ||||||
|  |         } | ||||||
|  |     ) | ||||||
| } | } | ||||||
|  |  | ||||||
| function callApi( |  | ||||||
|     endpoint: string, |  | ||||||
|     onSuccessHandler?: (response: AxiosResponse) => void, |  | ||||||
|     onErrorHandler?: (error: AxiosError) => void |  | ||||||
| ): void { |  | ||||||
|     axiosInstance |  | ||||||
|         .get(endpoint) |  | ||||||
|         .then( |  | ||||||
|             onSuccessHandler == null |  | ||||||
|                 ? defaultApiSuccessHandler |  | ||||||
|                 : onSuccessHandler |  | ||||||
|         ) |  | ||||||
|         .catch( |  | ||||||
|             onErrorHandler == null ? defaultApiErrorHandler : onErrorHandler |  | ||||||
|         ); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export { callApi }; | export {callApi} | ||||||
| @@ -1,31 +1,33 @@ | |||||||
| import Keycloak from "keycloak-js"; | import Keycloak from 'keycloak-js'; | ||||||
| import type { AuthStore } from "@/stores/authStore.ts"; |  | ||||||
|  |  | ||||||
| const options = { | const options = { | ||||||
|     url: import.meta.env.VITE_KEYCLOAK_URL, |     url: import.meta.env.VITE_KEYCLOAK_URL, | ||||||
|     clientId: import.meta.env.VITE_KEYCLOAK_CLIENT_ID, |     clientId: import.meta.env.VITE_KEYCLOAK_CLIENT_ID, | ||||||
|     realm: import.meta.env.VITE_KEYCLOAK_REALM, |     realm: import.meta.env.VITE_KEYCLOAK_REALM | ||||||
| }; | } | ||||||
|  |  | ||||||
|  |  | ||||||
| const keycloak = new Keycloak(options); | const keycloak = new Keycloak(options); | ||||||
| let authenticated: boolean | undefined; | let authenticated: boolean | undefined; | ||||||
| let store = null; | let store = null; | ||||||
|  |  | ||||||
| async function login() { | async function login(){ | ||||||
|     try { |     try { | ||||||
|         await keycloak.login(); // https://www.keycloak.org/securing-apps/javascript-adapter#:~:text=when%20initialization%20completes.-,login(options),-Redirects%20to%20login |         await keycloak.login() // https://www.keycloak.org/securing-apps/javascript-adapter#:~:text=when%20initialization%20completes.-,login(options),-Redirects%20to%20login | ||||||
|         return keycloak; |         return keycloak; | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.log(error); |         console.log(error) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| async function signup() { | async function signup(){ | ||||||
|     try { |     try { | ||||||
|         await keycloak.login({ action: "register" }); // https://www.keycloak.org/securing-apps/javascript-adapter#:~:text=when%20initialization%20completes.-,login(options),-Redirects%20to%20login |         await keycloak.login( | ||||||
|  |             {action: "register"} | ||||||
|  |         ) // https://www.keycloak.org/securing-apps/javascript-adapter#:~:text=when%20initialization%20completes.-,login(options),-Redirects%20to%20login | ||||||
|         return keycloak; |         return keycloak; | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.log(error); |         console.log(error) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -40,33 +42,31 @@ async function init(onInitCallback: () => void) { | |||||||
|             onLoad: "check-sso", |             onLoad: "check-sso", | ||||||
|             silentCheckSsoRedirectUri: `${location.origin}/silent-check-sso.htm`, |             silentCheckSsoRedirectUri: `${location.origin}/silent-check-sso.htm`, | ||||||
|             responseMode: "query", |             responseMode: "query", | ||||||
|         }); |         }) | ||||||
|         onInitCallback(); |         onInitCallback() | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.error("Keycloak init failed"); |         console.error("Keycloak init failed") | ||||||
|         console.error(error); |         console.error(error) | ||||||
|     } |     } | ||||||
| } | }; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Initializes store with Keycloak user data |  * Initializes store with Keycloak user data | ||||||
|  * |  * | ||||||
|  */ |  */ | ||||||
| async function initStore(storeInstance: AuthStore) { | async function initStore(storeInstance: any) { | ||||||
|     try { |     try { | ||||||
|         store = storeInstance; |         store = storeInstance | ||||||
|         console.log(keycloak); |         console.log(keycloak) | ||||||
|         await store.initOauth(keycloak); |         await store.initOauth(keycloak) | ||||||
|  |  | ||||||
|         // Show alert if user is not authenticated |         // Show alert if user is not authenticated | ||||||
|         if (!authenticated) { |         if (!authenticated) { console.warn("not authenticated") } | ||||||
|             console.warn("not authenticated"); |  | ||||||
|         } |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.error("Keycloak init failed"); |         console.error("Keycloak init failed") | ||||||
|         console.error(error); |         console.error(error) | ||||||
|     } |     } | ||||||
| } | }; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Logout user |  * Logout user | ||||||
| @@ -83,7 +83,7 @@ async function refreshToken() { | |||||||
|         await keycloak.updateToken(480); |         await keycloak.updateToken(480); | ||||||
|         return keycloak; |         return keycloak; | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.error("Failed to refresh token"); |         console.error('Failed to refresh token'); | ||||||
|         console.error(error); |         console.error(error); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -97,4 +97,4 @@ const KeycloakService = { | |||||||
|     callSignup: signup, |     callSignup: signup, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export default KeycloakService; | export default KeycloakService; | ||||||
| @@ -1,43 +1,19 @@ | |||||||
| import { ref, type Ref } from "vue"; | import {ref} from "vue"; | ||||||
|  | enum color {Red, Yellow, Blue, green} | ||||||
|  |  | ||||||
| enum color { | function addNewMessage(errorMessage: string, type?: color, timeout?: number){ | ||||||
|     Red, |     if (timeout == null){ | ||||||
|     Yellow, |  | ||||||
|     Blue, |  | ||||||
|     Green, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ErrorMessageContent = { |  | ||||||
|     message: string; |  | ||||||
|     color: color; |  | ||||||
|     id: number; |  | ||||||
|     timeout: number; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| let id: number = 0; |  | ||||||
| const getId = () => { |  | ||||||
|     id = id + 1; |  | ||||||
|     return id; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| function addNewMessage(errorMessage: string, type?: color, timeout?: number) { |  | ||||||
|     if (timeout == null) { |  | ||||||
|         timeout = 5000; |         timeout = 5000; | ||||||
|     } |     } | ||||||
|     if (type == null) { |     if (type == null){ | ||||||
|         type = color.Red; |         type = color.Red; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const data: ErrorMessageContent = { |     const data = {errorMessage: errorMessage, timeout: timeout, type: type, uid: Math.random()*100000}; | ||||||
|         message: errorMessage, |     errorList.value.push(data) | ||||||
|         timeout: timeout, |     setTimeout(() => errorList.value.slice(0, 1), timeout) | ||||||
|         color: type, |  | ||||||
|         id: getId(), |  | ||||||
|     }; |  | ||||||
|     errorList.value.push(data); |  | ||||||
|     setTimeout(() => errorList.value.slice(0, 1), timeout); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| const errorList: Ref<ErrorMessageContent[]> = ref([]); | const errorList: any= ref([]) | ||||||
|  |  | ||||||
| export { addNewMessage, errorList, color, type ErrorMessageContent }; | export {addNewMessage, errorList, color} | ||||||
| @@ -1,61 +1,54 @@ | |||||||
| import { defineStore } from "pinia"; | import { defineStore } from "pinia"; | ||||||
| import keycloakService from "@/services/keycloak"; | import keycloakService from '@/services/keycloak'; | ||||||
| import type Keycloak from "keycloak-js"; | import type Keycloak from "keycloak-js"; | ||||||
|  | export const useAuthStore = defineStore("storeAuth", { | ||||||
| const useAuthStore = defineStore("storeAuth", { |  | ||||||
|     state: () => { |     state: () => { | ||||||
|         return { |         return { | ||||||
|  |             testv: true, | ||||||
|             authenticated: false, |             authenticated: false, | ||||||
|             user: { |             user: { | ||||||
|                 token: "", |                 token: "", | ||||||
|                 refreshToken: "", |                 refreshToken: "", | ||||||
|                 username: "", |                 username: "", | ||||||
|             }, |             }, | ||||||
|         }; |         } | ||||||
|     }, |     }, | ||||||
|     persist: true, |     persist: true, | ||||||
|     getters: {}, |     getters: {}, | ||||||
|     actions: { |     actions: { | ||||||
|         // Initialize Keycloak OAuth |         // Initialize Keycloak OAuth | ||||||
|         async initOauth(keycloak: Keycloak, clearData = true) { |         async initOauth(keycloak: Keycloak, clearData = true) { | ||||||
|             if (clearData) { |             if(clearData) { await this.clearUserData(); } | ||||||
|                 await this.clearUserData(); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             this.authenticated = !!keycloak.authenticated; // the !! removes undefined |             this.authenticated = !!keycloak.authenticated; // the !! removes undefined | ||||||
|             if ( |             if (this.authenticated && keycloak.token && keycloak.idTokenParsed && keycloak.refreshToken){ | ||||||
|                 this.authenticated && |  | ||||||
|                 keycloak.token && |  | ||||||
|                 keycloak.idTokenParsed && |  | ||||||
|                 keycloak.refreshToken |  | ||||||
|             ) { |  | ||||||
|                 this.user.username = keycloak.idTokenParsed.given_name; |                 this.user.username = keycloak.idTokenParsed.given_name; | ||||||
|                 this.user.token = keycloak.token; |                 this.user.token = keycloak.token; | ||||||
|                 this.user.refreshToken = keycloak.refreshToken; |                 this.user.refreshToken = keycloak.refreshToken; | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         async login() { |         async login(){ | ||||||
|             try { |             try { | ||||||
|                 const keycloak = await keycloakService.callLogin(); |                 const keycloak = await keycloakService.callLogin(); | ||||||
|                 if (keycloak) await this.initOauth(keycloak); |                 if (keycloak) | ||||||
|  |                     await this.initOauth(keycloak); | ||||||
|             } catch (error) { |             } catch (error) { | ||||||
|                 console.log(error); |                 console.log(error) | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         async signup() { |         async signup() { | ||||||
|             try { |             try { | ||||||
|                 const keycloak = await keycloakService.callSignup(); |                 const keycloak = await keycloakService.callSignup(); | ||||||
|                 if (keycloak) await this.initOauth(keycloak); |                 if (keycloak) | ||||||
|  |                     await this.initOauth(keycloak); | ||||||
|             } catch (error) { |             } catch (error) { | ||||||
|                 console.log(error); |                 console.log(error) | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         // Logout user |         // Logout user | ||||||
|         async logout() { |         async logout() { | ||||||
|             try { |             try { | ||||||
|                 await keycloakService.CallLogout( |                 await keycloakService.CallLogout(import.meta.env.VITE_APP_URL + "/test"); | ||||||
|                     import.meta.env.VITE_APP_URL + "/test" |  | ||||||
|                 ); |  | ||||||
|                 await this.clearUserData(); |                 await this.clearUserData(); | ||||||
|             } catch (error) { |             } catch (error) { | ||||||
|                 console.error(error); |                 console.error(error); | ||||||
| @@ -65,23 +58,23 @@ const useAuthStore = defineStore("storeAuth", { | |||||||
|         async refreshUserToken() { |         async refreshUserToken() { | ||||||
|             try { |             try { | ||||||
|                 const keycloak = await keycloakService.CallTokenRefresh(); |                 const keycloak = await keycloakService.CallTokenRefresh(); | ||||||
|                 if (keycloak) await this.initOauth(keycloak, false); |                 if (keycloak) | ||||||
|  |                     await this.initOauth(keycloak, false); | ||||||
|             } catch (error) { |             } catch (error) { | ||||||
|                 console.error(error); |                 console.error(error); | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|  |         test() { | ||||||
|  |             this.testv = !this.testv; | ||||||
|  |         }, | ||||||
|         // Clear user's store data |         // Clear user's store data | ||||||
|         clearUserData() { |         clearUserData() { | ||||||
|             this.authenticated = false; |             this.authenticated = false; | ||||||
|             this.user = { |             this.user = { | ||||||
|                 token: "", |                 token: "", | ||||||
|                 refreshToken: "", |                     refreshToken: "", | ||||||
|                 username: "", |                     username: "", | ||||||
|             }; |             }; | ||||||
|         }, |         } | ||||||
|     }, |     } | ||||||
| }); | }); | ||||||
|  |  | ||||||
| type AuthStore = ReturnType<typeof useAuthStore>; |  | ||||||
|  |  | ||||||
| export { useAuthStore, type AuthStore }; |  | ||||||
							
								
								
									
										15
									
								
								front/MyINPulse-front/src/views/AboutView.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								front/MyINPulse-front/src/views/AboutView.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | <template> | ||||||
|  |   <div class="about"> | ||||||
|  |     <h1>This is an about page</h1> | ||||||
|  |   </div> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  | @media (min-width: 1024px) { | ||||||
|  |   .about { | ||||||
|  |     min-height: 100vh; | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | </style> | ||||||
| @@ -18,9 +18,9 @@ | |||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import Header from '../components/HeaderComponent.vue'; | import Header from '../components/Header.vue'; | ||||||
| import Agenda from "../components/Agenda.vue" | import Agenda from "../components/Agenda.vue" | ||||||
| import ProjectComp from '../components/ProjectComponent.vue'; | import ProjectComp from '../components/Project-comp.vue'; | ||||||
|  |  | ||||||
| import { ref } from "vue"; | import { ref } from "vue"; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,8 +11,7 @@ | |||||||
| </template> | </template> | ||||||
|    |    | ||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| // @ts-ignore | import HeaderCanvas from '../components/canvas/HeaderCanvas.vue'; | ||||||
| import HeaderCanvas from "../components/canvas/HeaderCanvas.vue"; |  | ||||||
| import LeanCanvas from '../components/canvas/LeanCanvas.vue'; | import LeanCanvas from '../components/canvas/LeanCanvas.vue'; | ||||||
| </script> | </script> | ||||||
|    |    | ||||||
| @@ -1,27 +1,22 @@ | |||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import { errorList } from "@/services/popupDisplayer.ts"; |  | ||||||
|  | import {errorList} from "@/services/popupDisplayer.ts"; | ||||||
| import ErrorModal from "@/components/errorModal.vue"; | import ErrorModal from "@/components/errorModal.vue"; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <template> | <template> | ||||||
|     <div class="error-wrapper"> | <div class="error-wrapper"> | ||||||
|         <error-modal |   <error-modal v-for="elm in errorList" :data=elm></error-modal> | ||||||
|             v-for="elm in errorList" | </div> | ||||||
|             :key="elm.id" |  | ||||||
|             :error-message="elm.message" |  | ||||||
|             :error-color="elm.color" |  | ||||||
|         /> |  | ||||||
|     </div> |  | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <style scoped> | <style scoped> | ||||||
|  |  | ||||||
| .error-wrapper{ | .error-wrapper{ | ||||||
|   position: absolute; |   position: absolute; | ||||||
|   left: 70%; |   left: 70%; | ||||||
|   /*background-color: blue;*/ |   //background-color: blue; | ||||||
|   height: 100%; |   height: 100%; | ||||||
|   width: 30%; |   width: 30%; | ||||||
|  |  | ||||||
| } | } | ||||||
| </style> | </style> | ||||||
							
								
								
									
										75
									
								
								front/MyINPulse-front/src/views/test.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								front/MyINPulse-front/src/views/test.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | |||||||
|  | <script setup lang="ts"> | ||||||
|  | import {store} from "../main.ts"; | ||||||
|  | import {callApi} from "@/services/api.ts"; | ||||||
|  | import ErrorModal from "@/components/errorModal.vue"; | ||||||
|  | import {errorList} from "@/services/popupDisplayer.ts"; | ||||||
|  | //import TempModal from "@/components/temp-modal.vue"; | ||||||
|  | import ErrorWrapper from "@/App.vue"; | ||||||
|  | function addResToTable(id: any){ | ||||||
|  |   return (req: any) => { | ||||||
|  |     console.log(req) | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <template> | ||||||
|  |   <h1>Test page</h1> | ||||||
|  |   <table class="test"  style="width:100%"> | ||||||
|  |     <tbody> | ||||||
|  |     <tr> | ||||||
|  |       <td>Is Currently Authenticated ? </td> | ||||||
|  |       <td>{{store.authenticated}}</td> | ||||||
|  |       <td><button @click="store.login">Login</button></td> | ||||||
|  |       <td><button @click="store.logout">Logout</button></td> | ||||||
|  |       <td><button @click="store.signup">Signup</button></td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |       <td>current token</td> | ||||||
|  |       <td>{{store.user.token}}</td> | ||||||
|  |       <td><button @click="store.refreshUserToken">Refresh</button></td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |       <td>Current refresh token</td> | ||||||
|  |       <td>{{store.user.refreshToken}}</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |       <td>Entrepreneur API call</td> | ||||||
|  |       <td><button @click="callApi('random')">call</button></td> | ||||||
|  |       <td>res</td> | ||||||
|  |       <td></td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |       <td>Admin API call</td> | ||||||
|  |       <td><button @click="callApi('random2')">call</button></td> | ||||||
|  |       <td>res</td> | ||||||
|  |       <td></td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |       <td>Unauth API call</td> | ||||||
|  |       <td><button @click="callApi('random3')">call</button></td> | ||||||
|  |       <td>res</td> | ||||||
|  |       <td id="3"></td> | ||||||
|  |     </tr> | ||||||
|  |     </tbody> | ||||||
|  |   </table> | ||||||
|  |   <temp-modal></temp-modal> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <style scoped> | ||||||
|  | table { | ||||||
|  |     width: 100px; | ||||||
|  |   table-layout: fixed | ||||||
|  |   } | ||||||
|  |   tr { | ||||||
|  |     width: 100%; | ||||||
|  |     height: 100%; | ||||||
|  |   } | ||||||
|  |   td { | ||||||
|  |     border: solid 1px black; | ||||||
|  |     width: 20%; | ||||||
|  |     height: 100%; | ||||||
|  |     overflow: hidden; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
| @@ -1,90 +0,0 @@ | |||||||
| <script lang="ts" setup> |  | ||||||
| import { store } from "../main.ts"; |  | ||||||
| import { callApi } from "@/services/api.ts"; |  | ||||||
| import { ref } from "vue"; |  | ||||||
|  |  | ||||||
| const CustomRequest = ref(""); |  | ||||||
| </script> |  | ||||||
|  |  | ||||||
| <template> |  | ||||||
|     <h1>Test page</h1> |  | ||||||
|     <table class="test" style="width: 100%"> |  | ||||||
|         <tbody> |  | ||||||
|             <tr> |  | ||||||
|                 <td>Is Currently Authenticated ?</td> |  | ||||||
|                 <td>{{ store.authenticated }}</td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="store.login">Login</button> |  | ||||||
|                 </td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="store.logout">Logout</button> |  | ||||||
|                 </td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="store.signup">Signup</button> |  | ||||||
|                 </td> |  | ||||||
|             </tr> |  | ||||||
|             <tr> |  | ||||||
|                 <td>current token</td> |  | ||||||
|                 <td>{{ store.user.token }}</td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="store.refreshUserToken">Refresh</button> |  | ||||||
|                 </td> |  | ||||||
|             </tr> |  | ||||||
|             <tr> |  | ||||||
|                 <td>Current refresh token</td> |  | ||||||
|                 <td>{{ store.user.refreshToken }}</td> |  | ||||||
|             </tr> |  | ||||||
|             <tr> |  | ||||||
|                 <td>Entrepreneur API call</td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="callApi('random')">call</button> |  | ||||||
|                 </td> |  | ||||||
|                 <td>res</td> |  | ||||||
|                 <td></td> |  | ||||||
|             </tr> |  | ||||||
|             <tr> |  | ||||||
|                 <td>Admin API call</td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="callApi('random2')">call</button> |  | ||||||
|                 </td> |  | ||||||
|                 <td>res</td> |  | ||||||
|                 <td></td> |  | ||||||
|             </tr> |  | ||||||
|             <tr> |  | ||||||
|                 <td>Unauth API call</td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="callApi('unauth/dev')">call</button> |  | ||||||
|                 </td> |  | ||||||
|                 <td>res</td> |  | ||||||
|                 <td id="3"></td> |  | ||||||
|             </tr> |  | ||||||
|             <tr> |  | ||||||
|                 <td> |  | ||||||
|                     <input v-model="CustomRequest" placeholder="edit me" /> |  | ||||||
|                 </td> |  | ||||||
|                 <td> |  | ||||||
|                     <button @click="callApi(CustomRequest)">call</button> |  | ||||||
|                 </td> |  | ||||||
|             </tr> |  | ||||||
|         </tbody> |  | ||||||
|     </table> |  | ||||||
| </template> |  | ||||||
|  |  | ||||||
| <style scoped> |  | ||||||
| table { |  | ||||||
|     width: 100px; |  | ||||||
|     table-layout: fixed; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| tr { |  | ||||||
|     width: 100%; |  | ||||||
|     height: 100%; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| td { |  | ||||||
|     border: solid 1px black; |  | ||||||
|     width: 20%; |  | ||||||
|     height: 100%; |  | ||||||
|     overflow: hidden; |  | ||||||
| } |  | ||||||
| </style> |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user