Compare commits
	
		
			275 Commits
		
	
	
		
			c5e7736a16
			...
			front_foun
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					47be7d340b | ||
| 
						 | 
					232d10b164 | ||
| bc7ce888ad | |||
| ed67a3734a | |||
| 95eb154556 | |||
| 19fef63b0e | |||
| 1fd95265ea | |||
| 
						 | 
					3ef2d8a198 | ||
| 
						 | 
					6b49bbbe57 | ||
| 
						 | 
					4c15cab607 | ||
| 
						 | 
					abfe92bc87 | ||
| 
						 | 
					85b4fe6a4c | ||
| f2448a029f | |||
| 
						 | 
					cef4daef15 | ||
| 
						 | 
					f5aba70017 | ||
| 
						 | 
					27adc81ddc | ||
| 
						 | 
					48f14e8a04 | ||
| d4533ea725 | |||
| 
						 | 
					7fc06035c7 | ||
| 
						 | 
					1b559f29b7 | ||
| 63327bc312 | |||
| 
						 | 
					f0cef41e2b | ||
| 7f16cdc86f | |||
| 72d6f49995 | |||
| 695ec5d9b8 | |||
| 0abafb4f7f | |||
| 3cd63e78e9 | |||
| 255af7ee7f | |||
| 3b308cfa6d | |||
| d039105f0a | |||
| 0a15dbbf2d | |||
| d1fce63ac5 | |||
| d9aaa225aa | |||
| d31bf259dd | |||
| 43b40c9432 | |||
| 
						 | 
					e84f69c21a | ||
| cc1fc9b45b | |||
| 
						 | 
					c76e83f2bf | ||
| 0d0ec255a5 | |||
| e0c43a5c95 | |||
| 
						 | 
					1f0f9196c4 | ||
| 40e577ef07 | |||
| 60302c44d2 | |||
| a6e4f80a01 | |||
| 02bff19de0 | |||
| ae36549de9 | |||
| b1a4c874ec | |||
| 829baac85e | |||
| 2e9d841709 | |||
| 25235f418a | |||
| 13845394e3 | |||
| 
						 | 
					73aac1875a | ||
| 
						 | 
					c3ad092512 | ||
| 
						 | 
					f4589c6306 | ||
| 
						 | 
					6004bce4e8 | ||
| 
						 | 
					0730275e75 | ||
| 4356a01e4a | |||
| 592331236e | |||
| 49e52e1826 | |||
| efcea7f680 | |||
| 8154814805 | |||
| 
						 | 
					7d41da97de | ||
| 
						 | 
					35f314498f | ||
| 
						 | 
					f46c3756f0 | ||
| 92696c3e16 | |||
| 5130c00796 | |||
| 5183a088e7 | |||
| 8ec569e6ff | |||
| b503cae235 | |||
| fcf4e1c01d | |||
| 
						 | 
					4f90da69f3 | ||
| 
						 | 
					0fe64fd844 | ||
| eb302268ba | |||
| 3f18304028 | |||
| bbb4debcd8 | |||
| 6f7fc70c4c | |||
| 
						 | 
					4044a95dd1 | ||
| aee8e8797c | |||
| 3d57ecb01a | |||
| 1f91ab72d8 | |||
| a4e13b0f0a | |||
| b116771375 | |||
| cbef042e97 | |||
| 63f6f16d83 | |||
| 864bbbb9fd | |||
| e890a03a48 | |||
| 351727f3d5 | |||
| b5637e4552 | |||
| 
						 | 
					8024b8070b | ||
| 
						 | 
					008e003207 | ||
| 820c979c94 | |||
| 2ebea35e5d | |||
| 
						 | 
					c3dded1e05 | ||
| 
						 | 
					8b48a55bf0 | ||
| 
						 | 
					32de0d6ca8 | ||
| 7534d0d9c6 | |||
| a0cbd5e324 | |||
| 
						 | 
					09eeaacfa6 | ||
| 
						 | 
					de09ab2891 | ||
| 6ae50f9cf7 | |||
| 0199e9d0dc | |||
| 05a56dc022 | |||
| 9f4596d0ba | |||
| 
						 | 
					be73815861 | ||
| 6d84b3ff93 | |||
| 
						 | 
					549028b1d0 | ||
| 
						 | 
					3de1ec71ff | ||
| 
						 | 
					a895b0afda | ||
| 
						 | 
					bb55209946 | ||
| 
						 | 
					a8e22de4a3 | ||
| f9ee12ab6f | |||
| 4d3aae1249 | |||
| 
						 | 
					fa96bd2b0d | ||
| 3673aa379c | |||
| b672b2e9f9 | |||
| 193876e51c | |||
| b8c7c6f587 | |||
| 
						 | 
					bee47473d5 | ||
| 
						 | 
					e7739af80b | ||
| 32557f8f87 | |||
| 8ee06b93a6 | |||
| fcc20d1f42 | |||
| 28b0e69da1 | |||
| 
						 | 
					6861d07dfc | ||
| 6226c9f632 | |||
| 7fce831c75 | |||
| 8403bc0592 | |||
| 5edcf9ffc8 | |||
| 5615b0fb11 | |||
| 8a13993d8a | |||
| 561f6d16b3 | |||
| 
						 | 
					7199e7ebbf | ||
| 
						 | 
					9762ca27fb | ||
| ca8c5d9209 | |||
| 4b6d501adc | |||
| 
						 | 
					08706af6c2 | ||
| 
						 | 
					01f062211a | ||
| bca88a7b20 | |||
| 
						 | 
					c60fb8945b | ||
| 
						 | 
					e20556ed0f | ||
| 
						 | 
					d9c5f7bacf | ||
| 
						 | 
					fdae3e4c04 | ||
| 36db3c2968 | |||
| a0eeb6715e | |||
| 
						 | 
					6d875d9df1 | ||
| 
						 | 
					7c8f3ba36a | ||
| 832539f43b | |||
| dfea20b9c4 | |||
| 
						 | 
					8b863ee4b1 | ||
| 
						 | 
					f96872fb6b | ||
| 
						 | 
					0140672812 | ||
| 
						 | 
					7df2c768c8 | ||
| 6b3cb2610d | |||
| 
						 | 
					6029457735 | ||
| ba99b3c2b0 | |||
| 31d82f6271 | |||
| 37e631a096 | |||
| 6306a00eca | |||
| 55112c8508 | |||
| 676f1204cb | |||
| 
						 | 
					c4ba7646d5 | ||
| 
						 | 
					4e1908d528 | ||
| 
						 | 
					03bbc77e8a | ||
| ad1fd45bed | |||
| f0c4a3a10d | |||
| 
						 | 
					70658e4fb9 | ||
| 
						 | 
					2b31753265 | ||
| f8991e90ab | |||
| 
						 | 
					66be0baca6 | ||
| 
						 | 
					60290956ec | ||
| 
						 | 
					b9647ce36e | ||
| 8c4b9ceb9d | |||
| 
						 | 
					84d8d4523b | ||
| 647812576e | |||
| 
						 | 
					2dfee66958 | ||
| 7e1271cfe2 | |||
| 801ecb3817 | |||
| 
						 | 
					cc89d4c79f | ||
| adf9a93e2e | |||
| 
						 | 
					37d8bcc719 | ||
| 
						 | 
					385c5cd8d0 | ||
| 
						 | 
					b672dd200c | ||
| 
						 | 
					9e1f568ea4 | ||
| 
						 | 
					aaa6e46d0c | ||
| 
						 | 
					2b1666c949 | ||
| 
						 | 
					0c724cae7f | ||
| 
						 | 
					6de45801d2 | ||
| 03897e1139 | |||
| 
						 | 
					9b9cfbdb2e | ||
| 00a733c03b | |||
| 
						 | 
					5b6b647697 | ||
| 3dc8131c33 | |||
| 
						 | 
					7c271d8c47 | ||
| 
						 | 
					81ce4fdb4c | ||
| 
						 | 
					ebd76a30ee | ||
| ead11215ba | |||
| 259d56271c | |||
| 
						 | 
					b9f3bbbe15 | ||
| 14a2a59786 | |||
| 0ae6e7dfda | |||
| 15ccb5630a | |||
| 4ec292cca7 | |||
| 14a953536a | |||
| 288f983816 | |||
| dd6032f3ef | |||
| 
						 | 
					7e2f5bc506 | ||
| 
						 | 
					4ef92efd0e | ||
| e769dd6757 | |||
| 
						 | 
					550a51523f | ||
| 323cb05388 | |||
| 
						 | 
					6ff6ce5052 | ||
| 
						 | 
					7e0851bfef | ||
| 
						 | 
					98b6d167e8 | ||
| 79baddb8f6 | |||
| 60ec920cff | |||
| 900a4c5bdc | |||
| d0b615c59d | |||
| 
						 | 
					0f8c83c2e2 | ||
| 
						 | 
					fad52644d2 | ||
| 
						 | 
					5f51a1008b | ||
| eccf116f49 | |||
| 8491c9b3cf | |||
| 
						 | 
					137bc84c21 | ||
| 
						 | 
					3c61fdca93 | ||
| 
						 | 
					5ee3755548 | ||
| 279c171ba2 | |||
| 9ba8e3e84e | |||
| 
						 | 
					52511dd4c4 | ||
| 6de38a9725 | |||
| 
						 | 
					84b70f8f38 | ||
| 
						 | 
					834d68949c | ||
| 
						 | 
					fea8687664 | ||
| 
						 | 
					c94d3654ce | ||
| 
						 | 
					d5c89bf8f4 | ||
| 
						 | 
					78c72bdd72 | ||
| 
						 | 
					307c7e700b | ||
| 
						 | 
					8d486dce89 | ||
| 
						 | 
					653f923693 | ||
| 
						 | 
					64da3c9ab0 | ||
| 
						 | 
					419ceec1bc | ||
| 
						 | 
					e011a5534e | ||
| 
						 | 
					ef964c4d35 | ||
| 
						 | 
					5608b12f84 | ||
| 
						 | 
					467babab79 | ||
| 
						 | 
					a2e2395cc2 | ||
| 
						 | 
					e3393c8834 | ||
| 
						 | 
					5f8fe4a374 | ||
| 067eeb9494 | |||
| 
						 | 
					f48b570494 | ||
| 
						 | 
					0733f8d5af | ||
| 
						 | 
					8071c01c5d | ||
| b355463dd9 | |||
| 4ee3d9bc44 | |||
| d75d45e204 | |||
| 79e949bdd4 | |||
| 
						 | 
					9f3754776f | ||
| 651fb2b1a1 | |||
| 
						 | 
					aa5988ce75 | ||
| 
						 | 
					9ae18e1e4b | ||
| ef8c8e896d | |||
| 22ebb0e1f4 | |||
| 09e4b3262f | |||
| 6a3d4239ab | |||
| 9d71c93b5b | |||
| b1df7421dc | |||
| 7a03146bf8 | |||
| 
						 | 
					5145b833ae | ||
| f0a371dc52 | |||
| 
						 | 
					ac19d33bdb | ||
| 
						 | 
					3d4d5b90d1 | ||
| 
						 | 
					4080cee818 | ||
| 
						 | 
					f4d73654d1 | ||
| 4fda5513a9 | |||
| 32407b0e8f | |||
| b30e1196f4 | 
@@ -9,6 +9,14 @@ jobs:
 | 
				
			|||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
    - name: Checkout sources
 | 
					    - name: Checkout sources
 | 
				
			||||||
      uses: actions/checkout@v4
 | 
					      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
 | 
					    - name: Setup Java
 | 
				
			||||||
      uses: actions/setup-java@v4
 | 
					      uses: actions/setup-java@v4
 | 
				
			||||||
      with:
 | 
					      with:
 | 
				
			||||||
@@ -18,8 +26,7 @@ jobs:
 | 
				
			|||||||
    - name: Setup Gradle
 | 
					    - name: Setup Gradle
 | 
				
			||||||
      uses: gradle/actions/setup-gradle@v4
 | 
					      uses: gradle/actions/setup-gradle@v4
 | 
				
			||||||
      with: 
 | 
					      with: 
 | 
				
			||||||
        cache-disabled: true
 | 
					        cache-disabled: true # Once the code has been pushed once in main, this should be reenabled. 
 | 
				
			||||||
 | 
					 | 
				
			||||||
    - name: init gradle
 | 
					    - name: init gradle
 | 
				
			||||||
      working-directory: ./MyINPulse-back/
 | 
					      working-directory: ./MyINPulse-back/
 | 
				
			||||||
      run: ./gradlew build -x test # todo: run test, currently fail because no database is present 
 | 
					      run: ./gradlew build # todo: run test, currently fail because no database is present 
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,5 +1,8 @@
 | 
				
			|||||||
.env
 | 
					.env
 | 
				
			||||||
.idea
 | 
					.idea
 | 
				
			||||||
keycloak/CAS/target
 | 
					keycloak/CAS/target
 | 
				
			||||||
 | 
					keycloak/.installed
 | 
				
			||||||
docker-compose.yaml
 | 
					docker-compose.yaml
 | 
				
			||||||
 | 
					node_modules
 | 
				
			||||||
 | 
					.vscode
 | 
				
			||||||
postgres/data
 | 
					postgres/data
 | 
				
			||||||
							
								
								
									
										24
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								Makefile
									
									
									
									
									
								
							@@ -19,8 +19,14 @@ front/MyINPulse-front/.installed:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
vite: ./front/MyINPulse-front/.installed
 | 
					vite: ./front/MyINPulse-front/.installed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					keycloak: ./keycloak/.installed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
dev-front: clean vite
 | 
					keycloak/.installed:
 | 
				
			||||||
 | 
						@echo "running one time install"
 | 
				
			||||||
 | 
						@cd keycloak/CAS && sudo sh build.sh
 | 
				
			||||||
 | 
						@touch ./keycloak/.installed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					dev-front: clean vite keycloak
 | 
				
			||||||
	@cp config/frontdev.env front/MyINPulse-front/.env
 | 
						@cp config/frontdev.env front/MyINPulse-front/.env
 | 
				
			||||||
	@cp config/frontdev.env .env
 | 
						@cp config/frontdev.env .env
 | 
				
			||||||
	@cp config/frontdev.env MyINPulse-back/.env
 | 
						@cp config/frontdev.env MyINPulse-back/.env
 | 
				
			||||||
@@ -28,7 +34,7 @@ dev-front: clean vite
 | 
				
			|||||||
	@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
 | 
					prod: clean keycloak
 | 
				
			||||||
	@cp config/prod.env front/MyINPulse-front/.env
 | 
						@cp config/prod.env front/MyINPulse-front/.env
 | 
				
			||||||
	@cp config/prod.env .env
 | 
						@cp config/prod.env .env
 | 
				
			||||||
	@cp config/prod.env .env
 | 
						@cp config/prod.env .env
 | 
				
			||||||
@@ -37,7 +43,7 @@ prod: clean
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
dev-back:
 | 
					dev-back: keycloak
 | 
				
			||||||
	@cp config/backdev.env front/MyINPulse-front/.env
 | 
						@cp config/backdev.env front/MyINPulse-front/.env
 | 
				
			||||||
	@cp config/backdev.env .env
 | 
						@cp config/backdev.env .env
 | 
				
			||||||
	@cp config/backdev.env MyINPulse-back/.env
 | 
						@cp config/backdev.env MyINPulse-back/.env
 | 
				
			||||||
@@ -46,7 +52,7 @@ dev-back:
 | 
				
			|||||||
	@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)'
 | 
						@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)'
 | 
				
			||||||
	@echo "./gradlew bootRun --args='--server.port=8081'"
 | 
						@echo "./gradlew bootRun --args='--server.port=8081'"
 | 
				
			||||||
	
 | 
						
 | 
				
			||||||
dev: clean vite
 | 
					dev: clean vite keycloak
 | 
				
			||||||
	@cp config/dev.env front/MyINPulse-front/.env
 | 
						@cp config/dev.env front/MyINPulse-front/.env
 | 
				
			||||||
	@cp config/dev.env .env
 | 
						@cp config/dev.env .env
 | 
				
			||||||
	@cp config/dev.env MyINPulse-back/.env
 | 
						@cp config/dev.env MyINPulse-back/.env
 | 
				
			||||||
@@ -55,3 +61,13 @@ dev: clean vite
 | 
				
			|||||||
	@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)'
 | 
						@echo "cd MyINPulse-back" && echo 'export $$(cat .env | xargs)'
 | 
				
			||||||
	@echo "./gradlew bootRun --args='--server.port=8081'"
 | 
						@echo "./gradlew bootRun --args='--server.port=8081'"
 | 
				
			||||||
	@cd ./front/MyINPulse-front/ && npm run dev &
 | 
						@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
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,6 +2,7 @@ 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'
 | 
				
			||||||
@@ -26,6 +27,9 @@ dependencies {
 | 
				
			|||||||
    implementation group: 'org.apache.logging.log4j', name: 'log4j-api', version: '2.+'
 | 
					    implementation group: 'org.apache.logging.log4j', name: 'log4j-api', version: '2.+'
 | 
				
			||||||
    implementation group: 'org.apache.logging.log4j', name: 'log4j-core', version: '2.+'
 | 
					    implementation group: 'org.apache.logging.log4j', name: 'log4j-core', version: '2.+'
 | 
				
			||||||
    implementation 'org.postgresql:postgresql'
 | 
					    implementation 'org.postgresql:postgresql'
 | 
				
			||||||
 | 
					    implementation group: 'com.itextpdf', name: 'itextpdf', version: '5.5.13.3'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    runtimeOnly 'com.h2database:h2'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    testImplementation 'org.springframework.boot:spring-boot-starter-test'
 | 
					    testImplementation 'org.springframework.boot:spring-boot-starter-test'
 | 
				
			||||||
    testImplementation 'com.h2database:h2'
 | 
					    testImplementation 'com.h2database:h2'
 | 
				
			||||||
@@ -36,3 +40,22 @@ dependencies {
 | 
				
			|||||||
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')
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -31,7 +31,7 @@ public class WebSecurityCustomConfiguration {
 | 
				
			|||||||
    public CorsConfigurationSource corsConfigurationSource() {
 | 
					    public CorsConfigurationSource corsConfigurationSource() {
 | 
				
			||||||
        CorsConfiguration configuration = new CorsConfiguration();
 | 
					        CorsConfiguration configuration = new CorsConfiguration();
 | 
				
			||||||
        configuration.setAllowedOrigins(List.of(frontendUrl));
 | 
					        configuration.setAllowedOrigins(List.of(frontendUrl));
 | 
				
			||||||
        configuration.setAllowedMethods(Arrays.asList("GET", "OPTIONS"));
 | 
					        configuration.setAllowedMethods(Arrays.asList("GET", "OPTIONS", "POST", "PUT", "DELETE"));
 | 
				
			||||||
        configuration.setAllowedHeaders(
 | 
					        configuration.setAllowedHeaders(
 | 
				
			||||||
                Arrays.asList("authorization", "content-type", "x-auth-token"));
 | 
					                Arrays.asList("authorization", "content-type", "x-auth-token"));
 | 
				
			||||||
        UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
 | 
					        UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
 | 
				
			||||||
@@ -56,14 +56,18 @@ public class WebSecurityCustomConfiguration {
 | 
				
			|||||||
        http.authorizeHttpRequests(
 | 
					        http.authorizeHttpRequests(
 | 
				
			||||||
                        authorize ->
 | 
					                        authorize ->
 | 
				
			||||||
                                authorize
 | 
					                                authorize
 | 
				
			||||||
                                        .requestMatchers("/entrepreneur/**", "/shared/**")
 | 
					                                        .requestMatchers("/entrepreneur/**")
 | 
				
			||||||
                                        .access(hasRole("REALM_MyINPulse-entrepreneur"))
 | 
					                                        .access(hasRole("REALM_MyINPulse-entrepreneur"))
 | 
				
			||||||
                                        .requestMatchers("/admin/**", "/shared/**")
 | 
					                                        .requestMatchers("/admin/**")
 | 
				
			||||||
                                        .access(hasRole("REALM_MyINPulse-admin"))
 | 
					                                        .access(hasRole("REALM_MyINPulse-admin"))
 | 
				
			||||||
 | 
					                                        .requestMatchers("/shared/**")
 | 
				
			||||||
 | 
					                                        .hasAnyRole(
 | 
				
			||||||
 | 
					                                                "REALM_MyINPulse-admin",
 | 
				
			||||||
 | 
					                                                "REALM_MyINPulse-entrepreneur")
 | 
				
			||||||
                                        .requestMatchers("/unauth/**")
 | 
					                                        .requestMatchers("/unauth/**")
 | 
				
			||||||
                                        .permitAll()
 | 
					                                        .authenticated()
 | 
				
			||||||
                                        .anyRequest()
 | 
					                                        .anyRequest()
 | 
				
			||||||
                                        .authenticated())
 | 
					                                        .denyAll())
 | 
				
			||||||
                .oauth2ResourceServer(
 | 
					                .oauth2ResourceServer(
 | 
				
			||||||
                        oauth2 ->
 | 
					                        oauth2 ->
 | 
				
			||||||
                                oauth2.jwt(
 | 
					                                oauth2.jwt(
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -57,7 +57,7 @@ public class AdminApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return the status code of the request
 | 
					     * @return the status code of the request
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PostMapping("/admin/projects/decision")
 | 
					    @PostMapping("/admin/projects/pending/decision")
 | 
				
			||||||
    public void validateProject(@RequestBody ProjectDecision decision) {
 | 
					    public void validateProject(@RequestBody ProjectDecision decision) {
 | 
				
			||||||
        adminApiService.validateProject(decision);
 | 
					        adminApiService.validateProject(decision);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -67,7 +67,7 @@ public class AdminApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return the status code of the request
 | 
					     * @return the status code of the request
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PostMapping("/admin/project/add")
 | 
					    @PostMapping("/admin/project")
 | 
				
			||||||
    public void addNewProject(@RequestBody Project project) {
 | 
					    public void addNewProject(@RequestBody Project project) {
 | 
				
			||||||
        adminApiService.addNewProject(project);
 | 
					        adminApiService.addNewProject(project);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -79,9 +79,9 @@ public class AdminApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return the status code of the request
 | 
					     * @return the status code of the request
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PostMapping("/admin/appoitements/report/{appointmentId}")
 | 
					    @PostMapping("/admin/appointments/report/{appointmentId}")
 | 
				
			||||||
    public void createAppointmentReport(
 | 
					    public void createAppointmentReport(
 | 
				
			||||||
            @PathVariable String appointmentId,
 | 
					            @PathVariable long appointmentId,
 | 
				
			||||||
            @RequestBody Report report,
 | 
					            @RequestBody Report report,
 | 
				
			||||||
            @AuthenticationPrincipal Jwt principal) {
 | 
					            @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        adminApiService.createAppointmentReport(
 | 
					        adminApiService.createAppointmentReport(
 | 
				
			||||||
@@ -95,8 +95,35 @@ public class AdminApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return the status code of the request
 | 
					     * @return the status code of the request
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @DeleteMapping("/admin/projects/remove/{projectId}")
 | 
					    @DeleteMapping("/admin/projects/{projectId}")
 | 
				
			||||||
    public void deleteProject(@PathVariable long projectId) {
 | 
					    public void deleteProject(@PathVariable long projectId) {
 | 
				
			||||||
        adminApiService.deleteProject(projectId);
 | 
					        adminApiService.deleteProject(projectId);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @PostMapping("/admin/make-admin/{userId}")
 | 
				
			||||||
 | 
					    public void setAdmin(@PathVariable long userId, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        this.adminApiService.setAdmin(userId, principal.getTokenValue());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @PostMapping("/admin/accounts/validate/{userId}")
 | 
				
			||||||
 | 
					    public void validateEntrepreneurAcc(
 | 
				
			||||||
 | 
					            @PathVariable long userId, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        this.adminApiService.validateEntrepreneurAccount(userId, principal.getTokenValue());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @GetMapping("/admin/pending-accounts")
 | 
				
			||||||
 | 
					    public Iterable<User> validateEntrepreneurAcc() {
 | 
				
			||||||
 | 
					        return this.adminApiService.getPendingUsers();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @PostMapping("/admin/create-account")
 | 
				
			||||||
 | 
					    public void createAccount(@AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        String userSurname = principal.getClaimAsString("userSurname");
 | 
				
			||||||
 | 
					        String username = principal.getClaimAsString("preferred_username");
 | 
				
			||||||
 | 
					        String primaryMail = principal.getClaimAsString("email");
 | 
				
			||||||
 | 
					        String secondaryMail = principal.getClaimAsString("secondaryMail");
 | 
				
			||||||
 | 
					        String phoneNumber = principal.getClaimAsString("phoneNumber");
 | 
				
			||||||
 | 
					        this.adminApiService.createAccount(
 | 
				
			||||||
 | 
					                userSurname, username, primaryMail, secondaryMail, phoneNumber);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,14 +1,20 @@
 | 
				
			|||||||
package enseirb.myinpulse.controller;
 | 
					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.beans.factory.annotation.Autowired;
 | 
				
			||||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
 | 
					import org.springframework.boot.autoconfigure.SpringBootApplication;
 | 
				
			||||||
import org.springframework.security.core.annotation.AuthenticationPrincipal;
 | 
					import org.springframework.security.core.annotation.AuthenticationPrincipal;
 | 
				
			||||||
import org.springframework.security.oauth2.jwt.Jwt;
 | 
					import org.springframework.security.oauth2.jwt.Jwt;
 | 
				
			||||||
import org.springframework.web.bind.annotation.*;
 | 
					import org.springframework.web.bind.annotation.DeleteMapping;
 | 
				
			||||||
 | 
					import org.springframework.web.bind.annotation.GetMapping;
 | 
				
			||||||
 | 
					import org.springframework.web.bind.annotation.PathVariable;
 | 
				
			||||||
 | 
					import org.springframework.web.bind.annotation.PostMapping;
 | 
				
			||||||
 | 
					import org.springframework.web.bind.annotation.PutMapping;
 | 
				
			||||||
 | 
					import org.springframework.web.bind.annotation.RequestBody;
 | 
				
			||||||
 | 
					import org.springframework.web.bind.annotation.RestController;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Project;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.SectionCell;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.EntrepreneurApiService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@SpringBootApplication
 | 
					@SpringBootApplication
 | 
				
			||||||
@RestController
 | 
					@RestController
 | 
				
			||||||
@@ -28,13 +34,26 @@ public class EntrepreneurApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return status code
 | 
					     * @return status code
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PutMapping("/entrepreneur/lcsection/modify/{sectionId}")
 | 
					    @PutMapping("/entrepreneur/sectionCells/{sectionCellId}")
 | 
				
			||||||
    public void editSectionCell(
 | 
					    public void editSectionCell(
 | 
				
			||||||
            @PathVariable Long sectionId,
 | 
					            @PathVariable Long sectionCellId,
 | 
				
			||||||
            @RequestBody SectionCell sectionCell,
 | 
					            @RequestBody String content,
 | 
				
			||||||
            @AuthenticationPrincipal Jwt principal) {
 | 
					            @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        entrepreneurApiService.editSectionCell(
 | 
					        entrepreneurApiService.editSectionCell(
 | 
				
			||||||
                sectionId, sectionCell, principal.getClaimAsString("email"));
 | 
					                sectionCellId, content, principal.getClaimAsString("email"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Endpoint used to update a LC section.
 | 
				
			||||||
 | 
					     *
 | 
				
			||||||
 | 
					     * @return status code
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @GetMapping("/entrepreneur/projects")
 | 
				
			||||||
 | 
					    public Iterable<Project> getEntrepreneurProjectId(
 | 
				
			||||||
 | 
					            @PathVariable Long sectionCellId,
 | 
				
			||||||
 | 
					            @RequestBody String content,
 | 
				
			||||||
 | 
					            @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        return entrepreneurApiService.getProjectIdViaClaim(principal.getClaimAsString("email"));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
@@ -44,10 +63,11 @@ public class EntrepreneurApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return status code
 | 
					     * @return status code
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @DeleteMapping("/entrepreneur/lcsection/remove/{sectionId}")
 | 
					    @DeleteMapping("/entrepreneur/sectionCells/{sectionCellId}")
 | 
				
			||||||
    public void removeSectionCell(
 | 
					    public void removeSectionCell(
 | 
				
			||||||
            @PathVariable Long sectionId, @AuthenticationPrincipal Jwt principal) {
 | 
					            @PathVariable Long sectionCellId, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        entrepreneurApiService.removeSectionCell(sectionId, principal.getClaimAsString("email"));
 | 
					        entrepreneurApiService.removeSectionCell(
 | 
				
			||||||
 | 
					                sectionCellId, principal.getClaimAsString("email"));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
@@ -57,7 +77,7 @@ public class EntrepreneurApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return status code
 | 
					     * @return status code
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PostMapping("/entrepreneur/lcsection/add") // remove id from doc aswell
 | 
					    @PostMapping("/entrepreneur/sectionCells")
 | 
				
			||||||
    public void addLCSection(
 | 
					    public void addLCSection(
 | 
				
			||||||
            @RequestBody SectionCell sectionCell, @AuthenticationPrincipal Jwt principal) {
 | 
					            @RequestBody SectionCell sectionCell, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        entrepreneurApiService.addSectionCell(sectionCell, principal.getClaimAsString("email"));
 | 
					        entrepreneurApiService.addSectionCell(sectionCell, principal.getClaimAsString("email"));
 | 
				
			||||||
@@ -70,9 +90,27 @@ public class EntrepreneurApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return status code
 | 
					     * @return status code
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PostMapping("/entrepreneur/project/request")
 | 
					    @PostMapping("/entrepreneur/projects/request")
 | 
				
			||||||
    public void requestNewProject(
 | 
					    public void requestNewProject(
 | 
				
			||||||
            @RequestBody Project project, @AuthenticationPrincipal Jwt principal) {
 | 
					            @RequestBody Project project, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        entrepreneurApiService.requestNewProject(project, principal.getClaimAsString("email"));
 | 
					        entrepreneurApiService.requestNewProject(project, principal.getClaimAsString("email"));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * <p>Endpoint to check if project is has already been validated by an admin
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @GetMapping("/entrepreneur/projects/project-is-active")
 | 
				
			||||||
 | 
					    public Boolean checkIfProjectValidated(@AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        return entrepreneurApiService.checkIfEntrepreneurProjectActive(
 | 
				
			||||||
 | 
					                principal.getClaimAsString("email"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * <p>Endpoint to check if a user requested a project (used when project is pending)
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @GetMapping("/entrepreneur/projects/has-pending-request")
 | 
				
			||||||
 | 
					    public Boolean checkIfHasRequested(@AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        return entrepreneurApiService.entrepreneurHasPendingRequestedProject(
 | 
				
			||||||
 | 
					                principal.getClaimAsString("email"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,7 @@
 | 
				
			|||||||
package enseirb.myinpulse.controller;
 | 
					package enseirb.myinpulse.controller;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.itextpdf.text.DocumentException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.*;
 | 
					import enseirb.myinpulse.model.*;
 | 
				
			||||||
import enseirb.myinpulse.service.SharedApiService;
 | 
					import enseirb.myinpulse.service.SharedApiService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -9,6 +11,9 @@ import org.springframework.security.core.annotation.AuthenticationPrincipal;
 | 
				
			|||||||
import org.springframework.security.oauth2.jwt.Jwt;
 | 
					import org.springframework.security.oauth2.jwt.Jwt;
 | 
				
			||||||
import org.springframework.web.bind.annotation.*;
 | 
					import org.springframework.web.bind.annotation.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					import java.net.URISyntaxException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@SpringBootApplication
 | 
					@SpringBootApplication
 | 
				
			||||||
@RestController
 | 
					@RestController
 | 
				
			||||||
public class SharedApi {
 | 
					public class SharedApi {
 | 
				
			||||||
@@ -25,7 +30,7 @@ public class SharedApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return a list of lean canvas sections
 | 
					     * @return a list of lean canvas sections
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @GetMapping("/shared/project/lcsection/{projectId}/{sectionId}/{date}")
 | 
					    @GetMapping("/shared/projects/sectionCells/{projectId}/{sectionId}/{date}")
 | 
				
			||||||
    public Iterable<SectionCell> getLCSection(
 | 
					    public Iterable<SectionCell> getLCSection(
 | 
				
			||||||
            @PathVariable("projectId") Long projectId,
 | 
					            @PathVariable("projectId") Long projectId,
 | 
				
			||||||
            @PathVariable("sectionId") Long sectionId,
 | 
					            @PathVariable("sectionId") Long sectionId,
 | 
				
			||||||
@@ -40,7 +45,7 @@ public class SharedApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return a list of all entrepreneurs in a project
 | 
					     * @return a list of all entrepreneurs in a project
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @GetMapping("/shared/entrepreneurs/{projectId}")
 | 
					    @GetMapping("/shared/projects/entrepreneurs/{projectId}")
 | 
				
			||||||
    public Iterable<Entrepreneur> getEntrepreneursByProjectId(
 | 
					    public Iterable<Entrepreneur> getEntrepreneursByProjectId(
 | 
				
			||||||
            @PathVariable int projectId, @AuthenticationPrincipal Jwt principal) {
 | 
					            @PathVariable int projectId, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        return sharedApiService.getEntrepreneursByProjectId(
 | 
					        return sharedApiService.getEntrepreneursByProjectId(
 | 
				
			||||||
@@ -75,16 +80,24 @@ public class SharedApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @return a PDF file? TODO: how does that works ?
 | 
					     * @return a PDF file? TODO: how does that works ?
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @GetMapping("/shared/projects/appointments/report/{appointmentId}")
 | 
					    @GetMapping("/shared/appointments/report/{appointmentId}")
 | 
				
			||||||
    public void getPDFReport(
 | 
					    public void getPDFReport(
 | 
				
			||||||
            @PathVariable int appointmentId, @AuthenticationPrincipal Jwt principal) {
 | 
					            @PathVariable int appointmentId, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        sharedApiService.getPDFReport(appointmentId, principal.getClaimAsString("email"));
 | 
					        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
 | 
					     * @return TODO
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    @PostMapping("/shared/appointment/request")
 | 
					    @PostMapping("/shared/appointments/request")
 | 
				
			||||||
    public void createAppointmentRequest(
 | 
					    public void createAppointmentRequest(
 | 
				
			||||||
            @RequestBody Appointment appointment, @AuthenticationPrincipal Jwt principal) {
 | 
					            @RequestBody Appointment appointment, @AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
        sharedApiService.createAppointmentRequest(appointment, principal.getClaimAsString("email"));
 | 
					        sharedApiService.createAppointmentRequest(appointment, principal.getClaimAsString("email"));
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1,61 @@
 | 
				
			|||||||
 | 
					package enseirb.myinpulse.controller;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Entrepreneur;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.EntrepreneurApiService;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.UtilsService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					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 UnauthApi {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private final EntrepreneurApiService entrepreneurApiService;
 | 
				
			||||||
 | 
					    private final UtilsService utilsService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Autowired
 | 
				
			||||||
 | 
					    UnauthApi(EntrepreneurApiService entrepreneurApiService, UtilsService utilsService) {
 | 
				
			||||||
 | 
					        this.entrepreneurApiService = entrepreneurApiService;
 | 
				
			||||||
 | 
					        this.utilsService = utilsService;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @PostMapping("/unauth/finalize")
 | 
				
			||||||
 | 
					    public void createAccount(@AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        boolean sneeStatus;
 | 
				
			||||||
 | 
					        if (principal.getClaimAsString("sneeStatus") != null) {
 | 
				
			||||||
 | 
					            sneeStatus = principal.getClaimAsString("sneeStatus").equals("true");
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            sneeStatus = false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        String userSurname = principal.getClaimAsString("userSurname");
 | 
				
			||||||
 | 
					        String username = principal.getClaimAsString("preferred_username");
 | 
				
			||||||
 | 
					        String primaryMail = principal.getClaimAsString("email");
 | 
				
			||||||
 | 
					        String secondaryMail = principal.getClaimAsString("secondaryMail");
 | 
				
			||||||
 | 
					        String phoneNumber = principal.getClaimAsString("phoneNumber");
 | 
				
			||||||
 | 
					        String school = principal.getClaimAsString("school");
 | 
				
			||||||
 | 
					        String course = principal.getClaimAsString("course");
 | 
				
			||||||
 | 
					        Entrepreneur e =
 | 
				
			||||||
 | 
					                new Entrepreneur(
 | 
				
			||||||
 | 
					                        userSurname,
 | 
				
			||||||
 | 
					                        username,
 | 
				
			||||||
 | 
					                        primaryMail,
 | 
				
			||||||
 | 
					                        secondaryMail,
 | 
				
			||||||
 | 
					                        phoneNumber,
 | 
				
			||||||
 | 
					                        school,
 | 
				
			||||||
 | 
					                        course,
 | 
				
			||||||
 | 
					                        sneeStatus,
 | 
				
			||||||
 | 
					                        true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        entrepreneurApiService.createAccount(e);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @GetMapping("/unauth/check-if-not-pending")
 | 
				
			||||||
 | 
					    public Boolean checkAccountStatus(@AuthenticationPrincipal Jwt principal) {
 | 
				
			||||||
 | 
					        // Throws 404 if user not found
 | 
				
			||||||
 | 
					        return utilsService.checkEntrepreneurNotPending(principal.getClaimAsString("email"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -34,9 +34,33 @@ public class Administrator extends User {
 | 
				
			|||||||
    public Administrator(
 | 
					    public Administrator(
 | 
				
			||||||
            String userSurname,
 | 
					            String userSurname,
 | 
				
			||||||
            String username,
 | 
					            String username,
 | 
				
			||||||
            String mainMail,
 | 
					            String primaryMail,
 | 
				
			||||||
            String secondaryMail,
 | 
					            String secondaryMail,
 | 
				
			||||||
            String phoneNumber) {
 | 
					            String phoneNumber) {
 | 
				
			||||||
        super(null, userSurname, username, mainMail, secondaryMail, phoneNumber);
 | 
					        super(userSurname, username, primaryMail, secondaryMail, phoneNumber, false);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    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;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -34,4 +34,28 @@ public class Annotation {
 | 
				
			|||||||
    public void setComment(String comment) {
 | 
					    public void setComment(String comment) {
 | 
				
			||||||
        this.comment = 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;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -112,7 +112,15 @@ public class Appointment {
 | 
				
			|||||||
        return listSectionCell;
 | 
					        return listSectionCell;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateListSectionCell(SectionCell sectionCell) {
 | 
				
			||||||
 | 
					        listSectionCell.add(sectionCell);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Report getAppointmentReport() {
 | 
					    public Report getAppointmentReport() {
 | 
				
			||||||
        return report;
 | 
					        return report;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setAppointmentReport(Report report) {
 | 
				
			||||||
 | 
					        this.report = report;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -37,21 +37,58 @@ public class Entrepreneur extends User {
 | 
				
			|||||||
    public Entrepreneur() {}
 | 
					    public Entrepreneur() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Entrepreneur(
 | 
					    public Entrepreneur(
 | 
				
			||||||
            Long idUser,
 | 
					 | 
				
			||||||
            String userSurname,
 | 
					            String userSurname,
 | 
				
			||||||
            String username,
 | 
					            String username,
 | 
				
			||||||
            String mainMail,
 | 
					            String primaryMail,
 | 
				
			||||||
 | 
					            String secondaryMail,
 | 
				
			||||||
 | 
					            String phoneNumber,
 | 
				
			||||||
 | 
					            String school,
 | 
				
			||||||
 | 
					            String course,
 | 
				
			||||||
 | 
					            boolean sneeStatus,
 | 
				
			||||||
 | 
					            boolean pending) {
 | 
				
			||||||
 | 
					        super(userSurname, username, primaryMail, secondaryMail, phoneNumber, pending);
 | 
				
			||||||
 | 
					        this.school = school;
 | 
				
			||||||
 | 
					        this.course = course;
 | 
				
			||||||
 | 
					        this.sneeStatus = sneeStatus;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Entrepreneur(
 | 
				
			||||||
 | 
					            String userSurname,
 | 
				
			||||||
 | 
					            String username,
 | 
				
			||||||
 | 
					            String primaryMail,
 | 
				
			||||||
            String secondaryMail,
 | 
					            String secondaryMail,
 | 
				
			||||||
            String phoneNumber,
 | 
					            String phoneNumber,
 | 
				
			||||||
            String school,
 | 
					            String school,
 | 
				
			||||||
            String course,
 | 
					            String course,
 | 
				
			||||||
            boolean sneeStatus) {
 | 
					            boolean sneeStatus) {
 | 
				
			||||||
        super(idUser, userSurname, username, mainMail, secondaryMail, phoneNumber);
 | 
					        super(userSurname, username, primaryMail, secondaryMail, phoneNumber, true);
 | 
				
			||||||
        this.school = school;
 | 
					        this.school = school;
 | 
				
			||||||
        this.course = course;
 | 
					        this.course = course;
 | 
				
			||||||
        this.sneeStatus = sneeStatus;
 | 
					        this.sneeStatus = sneeStatus;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Entrepreneur(
 | 
				
			||||||
 | 
					            String userSurname,
 | 
				
			||||||
 | 
					            String userName,
 | 
				
			||||||
 | 
					            String primaryMail,
 | 
				
			||||||
 | 
					            String secondaryMail,
 | 
				
			||||||
 | 
					            String phoneNumber,
 | 
				
			||||||
 | 
					            String school,
 | 
				
			||||||
 | 
					            String course,
 | 
				
			||||||
 | 
					            boolean sneeStatus,
 | 
				
			||||||
 | 
					            Project projectParticipation,
 | 
				
			||||||
 | 
					            Project projectProposed,
 | 
				
			||||||
 | 
					            MakeAppointment makeAppointment,
 | 
				
			||||||
 | 
					            boolean pending) {
 | 
				
			||||||
 | 
					        super(userSurname, userName, primaryMail, secondaryMail, phoneNumber, pending);
 | 
				
			||||||
 | 
					        this.school = school;
 | 
				
			||||||
 | 
					        this.course = course;
 | 
				
			||||||
 | 
					        this.sneeStatus = sneeStatus;
 | 
				
			||||||
 | 
					        this.projectParticipation = projectParticipation;
 | 
				
			||||||
 | 
					        this.projectProposed = projectProposed;
 | 
				
			||||||
 | 
					        this.makeAppointment = makeAppointment;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public String getSchool() {
 | 
					    public String getSchool() {
 | 
				
			||||||
        return school;
 | 
					        return school;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -79,4 +116,24 @@ public class Entrepreneur extends User {
 | 
				
			|||||||
    public Project getProjectParticipation() {
 | 
					    public Project getProjectParticipation() {
 | 
				
			||||||
        return projectParticipation;
 | 
					        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;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -26,8 +26,7 @@ public class Project {
 | 
				
			|||||||
    private byte[] logo;
 | 
					    private byte[] logo;
 | 
				
			||||||
    private LocalDate creationDate;
 | 
					    private LocalDate creationDate;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Column(length = 255)
 | 
					    @Column private ProjectDecisionValue projectStatus;
 | 
				
			||||||
    private String projectStatus;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @ManyToOne(fetch = FetchType.LAZY)
 | 
					    @ManyToOne(fetch = FetchType.LAZY)
 | 
				
			||||||
    @JoinColumn(name = "idAdministrator")
 | 
					    @JoinColumn(name = "idAdministrator")
 | 
				
			||||||
@@ -39,16 +38,41 @@ public class Project {
 | 
				
			|||||||
    public Project() {}
 | 
					    public Project() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Project(
 | 
					    public Project(
 | 
				
			||||||
            Long idProject,
 | 
					 | 
				
			||||||
            String projectName,
 | 
					            String projectName,
 | 
				
			||||||
            byte[] logo,
 | 
					            byte[] logo,
 | 
				
			||||||
            LocalDate creationDate,
 | 
					            LocalDate creationDate,
 | 
				
			||||||
            String projectStatus) {
 | 
					            ProjectDecisionValue projectStatus,
 | 
				
			||||||
        this.idProject = idProject;
 | 
					            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.projectName = projectName;
 | 
				
			||||||
        this.logo = logo;
 | 
					        this.logo = logo;
 | 
				
			||||||
        this.creationDate = creationDate;
 | 
					        this.creationDate = creationDate;
 | 
				
			||||||
        this.projectStatus = projectStatus;
 | 
					        this.projectStatus = projectStatus;
 | 
				
			||||||
 | 
					        this.projectAdministrator = projectAdministrator;
 | 
				
			||||||
 | 
					        this.entrepreneurProposed = entrepreneurProposed;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    public boolean equals(Object o) {
 | 
				
			||||||
 | 
					        if (o == this) {
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Project project = (Project) o;
 | 
				
			||||||
 | 
					        return this.idProject == project.idProject;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Long getIdProject() {
 | 
					    public Long getIdProject() {
 | 
				
			||||||
@@ -83,19 +107,43 @@ public class Project {
 | 
				
			|||||||
        this.creationDate = creationDate;
 | 
					        this.creationDate = creationDate;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public String getProjectStatus() {
 | 
					    public ProjectDecisionValue getProjectStatus() {
 | 
				
			||||||
        return projectStatus;
 | 
					        return projectStatus;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void setProjectStatus(String projectStatus) {
 | 
					    public void setProjectStatus(ProjectDecisionValue projectStatus) {
 | 
				
			||||||
        this.projectStatus = projectStatus;
 | 
					        this.projectStatus = projectStatus;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Administrator getAdministrator() {
 | 
					    public List<Entrepreneur> getListEntrepreneurParticipation() {
 | 
				
			||||||
        return this.projectAdministrator;
 | 
					        return listEntrepreneurParticipation;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void setAdministrator(Administrator administrator) {
 | 
					    public void updateListEntrepreneurParticipation(Entrepreneur projectParticipant) {
 | 
				
			||||||
        this.projectAdministrator = administrator;
 | 
					        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;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -4,4 +4,22 @@ public class ProjectDecision {
 | 
				
			|||||||
    public long projectId;
 | 
					    public long projectId;
 | 
				
			||||||
    public long adminId;
 | 
					    public long adminId;
 | 
				
			||||||
    public long isAccepted;
 | 
					    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
 | 
				
			||||||
 | 
					                + '}';
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					package enseirb.myinpulse.model;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public enum ProjectDecisionValue {
 | 
				
			||||||
 | 
					    PENDING,
 | 
				
			||||||
 | 
					    ACTIVE,
 | 
				
			||||||
 | 
					    ENDED,
 | 
				
			||||||
 | 
					    ABORTED,
 | 
				
			||||||
 | 
					    REJECTED,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -37,4 +37,12 @@ public class Report {
 | 
				
			|||||||
    public void setReportContent(String reportContent) {
 | 
					    public void setReportContent(String reportContent) {
 | 
				
			||||||
        this.reportContent = reportContent;
 | 
					        this.reportContent = reportContent;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Appointment getAppointmentReport() {
 | 
				
			||||||
 | 
					        return appointmentReport;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setAppointmentReport(Appointment appointmentReport) {
 | 
				
			||||||
 | 
					        this.appointmentReport = appointmentReport;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,6 +2,9 @@ package enseirb.myinpulse.model;
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import jakarta.persistence.*;
 | 
					import jakarta.persistence.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.hibernate.annotations.Generated;
 | 
				
			||||||
 | 
					import org.hibernate.generator.EventType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import java.time.LocalDateTime;
 | 
					import java.time.LocalDateTime;
 | 
				
			||||||
import java.util.ArrayList;
 | 
					import java.util.ArrayList;
 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
@@ -11,7 +14,7 @@ import java.util.List;
 | 
				
			|||||||
public class SectionCell {
 | 
					public class SectionCell {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @ManyToMany(mappedBy = "listSectionCell")
 | 
					    @ManyToMany(mappedBy = "listSectionCell")
 | 
				
			||||||
    private final List<Appointment> appointment = new ArrayList<>();
 | 
					    private final List<Appointment> listAppointment = new ArrayList<>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @OneToMany(mappedBy = "sectionCellAnnotation", fetch = FetchType.LAZY, orphanRemoval = true)
 | 
					    @OneToMany(mappedBy = "sectionCellAnnotation", fetch = FetchType.LAZY, orphanRemoval = true)
 | 
				
			||||||
    private final List<Annotation> listAnnotation = new ArrayList<>();
 | 
					    private final List<Annotation> listAnnotation = new ArrayList<>();
 | 
				
			||||||
@@ -20,6 +23,10 @@ public class SectionCell {
 | 
				
			|||||||
    @GeneratedValue(strategy = GenerationType.IDENTITY)
 | 
					    @GeneratedValue(strategy = GenerationType.IDENTITY)
 | 
				
			||||||
    private Long idSectionCell;
 | 
					    private Long idSectionCell;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Column(columnDefinition = "serial")
 | 
				
			||||||
 | 
					    @Generated(event = EventType.INSERT)
 | 
				
			||||||
 | 
					    private Long idReference;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Column() private long sectionId;
 | 
					    @Column() private long sectionId;
 | 
				
			||||||
    private String contentSectionCell;
 | 
					    private String contentSectionCell;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -39,11 +46,13 @@ public class SectionCell {
 | 
				
			|||||||
            Long idSectionCell,
 | 
					            Long idSectionCell,
 | 
				
			||||||
            Long sectionId,
 | 
					            Long sectionId,
 | 
				
			||||||
            String contentSectionCell,
 | 
					            String contentSectionCell,
 | 
				
			||||||
            LocalDateTime modificationDate) {
 | 
					            LocalDateTime modificationDate,
 | 
				
			||||||
 | 
					            Project projectSectionCell) {
 | 
				
			||||||
        this.idSectionCell = idSectionCell;
 | 
					        this.idSectionCell = idSectionCell;
 | 
				
			||||||
        this.sectionId = sectionId;
 | 
					        this.sectionId = sectionId;
 | 
				
			||||||
        this.contentSectionCell = contentSectionCell;
 | 
					        this.contentSectionCell = contentSectionCell;
 | 
				
			||||||
        this.modificationDate = modificationDate;
 | 
					        this.modificationDate = modificationDate;
 | 
				
			||||||
 | 
					        this.projectSectionCell = projectSectionCell;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Long getIdSectionCell() {
 | 
					    public Long getIdSectionCell() {
 | 
				
			||||||
@@ -54,6 +63,14 @@ public class SectionCell {
 | 
				
			|||||||
        this.idSectionCell = idSectionCell;
 | 
					        this.idSectionCell = idSectionCell;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Long getIdReference() {
 | 
				
			||||||
 | 
					        return idReference;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setIdReference(Long idReference) {
 | 
				
			||||||
 | 
					        this.idReference = idReference;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Long getSectionId() {
 | 
					    public Long getSectionId() {
 | 
				
			||||||
        return sectionId;
 | 
					        return sectionId;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -83,6 +100,26 @@ public class SectionCell {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public List<Appointment> getAppointmentSectionCell() {
 | 
					    public List<Appointment> getAppointmentSectionCell() {
 | 
				
			||||||
        return appointment;
 | 
					        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;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -26,21 +26,23 @@ public class User {
 | 
				
			|||||||
    @Column(length = 20)
 | 
					    @Column(length = 20)
 | 
				
			||||||
    private String phoneNumber;
 | 
					    private String phoneNumber;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Column private boolean pending;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public User() {}
 | 
					    public User() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public User(
 | 
					    public User(
 | 
				
			||||||
            Long idUser,
 | 
					 | 
				
			||||||
            String userSurname,
 | 
					            String userSurname,
 | 
				
			||||||
            String userName,
 | 
					            String userName,
 | 
				
			||||||
            String primaryMail,
 | 
					            String primaryMail,
 | 
				
			||||||
            String secondaryMail,
 | 
					            String secondaryMail,
 | 
				
			||||||
            String phoneNumber) {
 | 
					            String phoneNumber,
 | 
				
			||||||
        this.idUser = idUser;
 | 
					            boolean pending) {
 | 
				
			||||||
        this.userSurname = userSurname;
 | 
					        this.userSurname = userSurname;
 | 
				
			||||||
        this.userName = userName;
 | 
					        this.userName = userName;
 | 
				
			||||||
        this.primaryMail = primaryMail;
 | 
					        this.primaryMail = primaryMail;
 | 
				
			||||||
        this.secondaryMail = secondaryMail;
 | 
					        this.secondaryMail = secondaryMail;
 | 
				
			||||||
        this.phoneNumber = phoneNumber;
 | 
					        this.phoneNumber = phoneNumber;
 | 
				
			||||||
 | 
					        this.pending = pending;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Long getIdUser() {
 | 
					    public Long getIdUser() {
 | 
				
			||||||
@@ -71,8 +73,8 @@ public class User {
 | 
				
			|||||||
        return primaryMail;
 | 
					        return primaryMail;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void setPrimaryMail(String mainMail) {
 | 
					    public void setPrimaryMail(String primaryMail) {
 | 
				
			||||||
        this.primaryMail = mainMail;
 | 
					        this.primaryMail = primaryMail;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public String getSecondaryMail() {
 | 
					    public String getSecondaryMail() {
 | 
				
			||||||
@@ -90,4 +92,12 @@ public class User {
 | 
				
			|||||||
    public void setPhoneNumber(String phoneNumber) {
 | 
					    public void setPhoneNumber(String phoneNumber) {
 | 
				
			||||||
        phoneNumber = phoneNumber;
 | 
					        phoneNumber = phoneNumber;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public boolean isPending() {
 | 
				
			||||||
 | 
					        return pending;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setPending(boolean pending) {
 | 
				
			||||||
 | 
					        this.pending = pending;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -5,10 +5,13 @@ import enseirb.myinpulse.model.Administrator;
 | 
				
			|||||||
import org.springframework.data.jpa.repository.JpaRepository;
 | 
					import org.springframework.data.jpa.repository.JpaRepository;
 | 
				
			||||||
import org.springframework.data.rest.core.annotation.RepositoryRestResource;
 | 
					import org.springframework.data.rest.core.annotation.RepositoryRestResource;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.Optional;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@RepositoryRestResource
 | 
					@RepositoryRestResource
 | 
				
			||||||
public interface AdministratorRepository extends JpaRepository<Administrator, Long> {
 | 
					public interface AdministratorRepository extends JpaRepository<Administrator, Long> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* @Query("SELECT a from Administrators a")
 | 
					    /* @Query("SELECT a from Administrators a")
 | 
				
			||||||
    Administrator findAllAdministrator(); */
 | 
					    Administrator findAllAdministrator(); */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Optional<Administrator> findByPrimaryMail(String PrimaryMail);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -2,13 +2,18 @@ package enseirb.myinpulse.repository;
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.Administrator;
 | 
					import enseirb.myinpulse.model.Administrator;
 | 
				
			||||||
import enseirb.myinpulse.model.Project;
 | 
					import enseirb.myinpulse.model.Project;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.ProjectDecisionValue;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.springframework.data.jpa.repository.JpaRepository;
 | 
					import org.springframework.data.jpa.repository.JpaRepository;
 | 
				
			||||||
import org.springframework.data.rest.core.annotation.RepositoryRestResource;
 | 
					import org.springframework.data.rest.core.annotation.RepositoryRestResource;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.Optional;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@RepositoryRestResource
 | 
					@RepositoryRestResource
 | 
				
			||||||
public interface ProjectRepository extends JpaRepository<Project, Long> {
 | 
					public interface ProjectRepository extends JpaRepository<Project, Long> {
 | 
				
			||||||
    Iterable<Project> findByProjectAdministrator(Administrator administrator);
 | 
					    Iterable<Project> findByProjectAdministrator(Administrator administrator);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    Iterable<Project> findByProjectStatus(String status);
 | 
					    Iterable<Project> findByProjectStatus(ProjectDecisionValue status);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Optional<Project> findByProjectName(String projectName);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -15,4 +15,6 @@ public interface SectionCellRepository extends JpaRepository<SectionCell, Long>
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    Iterable<SectionCell> findByProjectSectionCellAndSectionIdAndModificationDateBefore(
 | 
					    Iterable<SectionCell> findByProjectSectionCellAndSectionIdAndModificationDateBefore(
 | 
				
			||||||
            Project project, long sectionId, LocalDateTime date);
 | 
					            Project project, long sectionId, LocalDateTime date);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Iterable<SectionCell> findByProjectSectionCell(Project project);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,8 +9,9 @@ import java.util.Optional;
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
@RepositoryRestResource
 | 
					@RepositoryRestResource
 | 
				
			||||||
public interface UserRepository extends JpaRepository<User, Long> {
 | 
					public interface UserRepository extends JpaRepository<User, Long> {
 | 
				
			||||||
    Optional<User> findByPrimaryMail(String email);
 | 
					    Optional<User> findByPrimaryMail(String primaryMail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Iterable<User> findAllByPendingEquals(boolean pending);
 | 
				
			||||||
    /* @Query("SELECT u from User u")
 | 
					    /* @Query("SELECT u from User u")
 | 
				
			||||||
    User findAllUser(); */
 | 
					    User findAllUser(); */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,72 +1,224 @@
 | 
				
			|||||||
package enseirb.myinpulse.service;
 | 
					package enseirb.myinpulse.service;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.*;
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.ACTIVE;
 | 
				
			||||||
import enseirb.myinpulse.service.database.AdministratorService;
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.REJECTED;
 | 
				
			||||||
import enseirb.myinpulse.service.database.ProjectService;
 | 
					 | 
				
			||||||
import enseirb.myinpulse.service.database.UserService;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					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.beans.factory.annotation.Autowired;
 | 
				
			||||||
import org.springframework.http.HttpStatus;
 | 
					import org.springframework.http.HttpStatus;
 | 
				
			||||||
import org.springframework.stereotype.Service;
 | 
					import org.springframework.stereotype.Service;
 | 
				
			||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					import org.springframework.web.server.ResponseStatusException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Service
 | 
					@Service
 | 
				
			||||||
public class AdminApiService {
 | 
					public class AdminApiService {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    protected static final Logger logger = LogManager.getLogger();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    private final ProjectService projectService;
 | 
					    private final ProjectService projectService;
 | 
				
			||||||
    private final UserService userService;
 | 
					    private final UserService userService;
 | 
				
			||||||
    private final AdministratorService administratorService;
 | 
					    private final AdministratorService administratorService;
 | 
				
			||||||
 | 
					    private final EntrepreneurService entrepreneurService;
 | 
				
			||||||
 | 
					    private final UtilsService utilsService;
 | 
				
			||||||
 | 
					    private final AppointmentService appointmentService;
 | 
				
			||||||
 | 
					    private final ReportService reportService;
 | 
				
			||||||
 | 
					    private final SectionCellService sectionCellService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Autowired
 | 
					    @Autowired
 | 
				
			||||||
    AdminApiService(
 | 
					    AdminApiService(
 | 
				
			||||||
            ProjectService projectService,
 | 
					            ProjectService projectService,
 | 
				
			||||||
            UserService userService,
 | 
					            UserService userService,
 | 
				
			||||||
            AdministratorService administratorService) {
 | 
					            AdministratorService administratorService,
 | 
				
			||||||
 | 
					            UtilsService utilsService,
 | 
				
			||||||
 | 
					            EntrepreneurService entrepreneurService,
 | 
				
			||||||
 | 
					            AppointmentService appointmentService,
 | 
				
			||||||
 | 
					            ReportService reportService,
 | 
				
			||||||
 | 
					            SectionCellService sectionCellService) {
 | 
				
			||||||
        this.projectService = projectService;
 | 
					        this.projectService = projectService;
 | 
				
			||||||
        this.userService = userService;
 | 
					        this.userService = userService;
 | 
				
			||||||
        this.administratorService = administratorService;
 | 
					        this.administratorService = administratorService;
 | 
				
			||||||
 | 
					        this.utilsService = utilsService;
 | 
				
			||||||
 | 
					        this.appointmentService = appointmentService;
 | 
				
			||||||
 | 
					        this.reportService = reportService;
 | 
				
			||||||
 | 
					        this.sectionCellService = sectionCellService;
 | 
				
			||||||
 | 
					        this.entrepreneurService = entrepreneurService;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO: test
 | 
					    // TODO: check if tests are sufficient - peer verification required
 | 
				
			||||||
    public Iterable<Project> getProjectsOfAdmin(String email) {
 | 
					    public Iterable<Project> getProjectsOfAdmin(String mail) {
 | 
				
			||||||
        return projectService.getProjectsByAdminId(
 | 
					        return projectService.getProjectsByAdminId(
 | 
				
			||||||
                administratorService.getAdministratorById(
 | 
					                administratorService.getAdministratorById(
 | 
				
			||||||
                        this.userService.getUserByEmail(email).getIdUser()));
 | 
					                        this.userService.getUserByEmail(mail).getIdUser()));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO
 | 
					    public Iterable<Appointment> getUpcomingAppointments(String mail) {
 | 
				
			||||||
    public Iterable<Appointment> getUpcomingAppointments(String email) {
 | 
					        logger.info("User {} check their upcoming appointments", mail);
 | 
				
			||||||
        throw new ResponseStatusException(HttpStatus.NOT_IMPLEMENTED, "Not implemented yet");
 | 
					        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();
 | 
				
			||||||
 | 
					            if (project == null) {
 | 
				
			||||||
 | 
					                throw new ResponseStatusException(
 | 
				
			||||||
 | 
					                        HttpStatus.NOT_FOUND,
 | 
				
			||||||
 | 
					                        "The user has no project, thus no appointments. No users should have no project");
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            project.getListSectionCell()
 | 
				
			||||||
 | 
					                    .forEach(
 | 
				
			||||||
 | 
					                            sectionCell -> {
 | 
				
			||||||
 | 
					                                appointments.addAll(
 | 
				
			||||||
 | 
					                                        this.sectionCellService.getAppointmentsBySectionCellId(
 | 
				
			||||||
 | 
					                                                sectionCell.getIdSectionCell()));
 | 
				
			||||||
 | 
					                            });
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return appointments;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO: test
 | 
					    // TODO: check if tests are sufficient - peer verification required
 | 
				
			||||||
    public Iterable<Project> getPendingProjects() {
 | 
					    public Iterable<Project> getPendingProjects() {
 | 
				
			||||||
        return this.projectService.getPendingProjects();
 | 
					        return this.projectService.getPendingProjects();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO: test
 | 
					    // TODO: check if tests are sufficient - peer verification required
 | 
				
			||||||
    public void validateProject(ProjectDecision decision) {
 | 
					    public void validateProject(ProjectDecision decision) {
 | 
				
			||||||
        projectService.updateProject(
 | 
					        projectService.updateProject(
 | 
				
			||||||
                decision.projectId,
 | 
					                decision.projectId,
 | 
				
			||||||
                null,
 | 
					                null,
 | 
				
			||||||
                null,
 | 
					                null,
 | 
				
			||||||
 | 
					                (decision.isAccepted == 1) ? ACTIVE : REJECTED,
 | 
				
			||||||
                null,
 | 
					                null,
 | 
				
			||||||
                "ACTIVE",
 | 
					                null,
 | 
				
			||||||
                this.administratorService.getAdministratorById(decision.projectId));
 | 
					                this.administratorService.getAdministratorById(decision.adminId));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO: solve todo + test
 | 
					    // TODO: check if tests are sufficient - peer verification required
 | 
				
			||||||
    public void addNewProject(Project project) {
 | 
					    public Project addNewProject(Project project) {
 | 
				
			||||||
        projectService.addNewProject(project); // TODO: how can the front know the ID ?
 | 
					        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);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					        return newProject;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO
 | 
					    public void createAppointmentReport(long appointmentId, Report report, String mail) {
 | 
				
			||||||
    public void createAppointmentReport(String appointmentId, Report report, String email) {
 | 
					        long projectId =
 | 
				
			||||||
        throw new ResponseStatusException(HttpStatus.NOT_IMPLEMENTED, "Not implemented yet");
 | 
					                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
 | 
					    // TODO: test
 | 
				
			||||||
    public void deleteProject(long projectId) {
 | 
					    public void deleteProject(long projectId) {
 | 
				
			||||||
        this.projectService.deleteProjectById(projectId);
 | 
					        this.projectService.deleteProjectById(projectId);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setAdmin(long userId, String token) {
 | 
				
			||||||
 | 
					        Entrepreneur e = this.entrepreneurService.getEntrepreneurById(userId);
 | 
				
			||||||
 | 
					        Administrator a =
 | 
				
			||||||
 | 
					                new Administrator(
 | 
				
			||||||
 | 
					                        e.getUserSurname(),
 | 
				
			||||||
 | 
					                        e.getUserName(),
 | 
				
			||||||
 | 
					                        e.getPrimaryMail(),
 | 
				
			||||||
 | 
					                        e.getSecondaryMail(),
 | 
				
			||||||
 | 
					                        e.getPhoneNumber());
 | 
				
			||||||
 | 
					        this.entrepreneurService.deleteEntrepreneur(e);
 | 
				
			||||||
 | 
					        this.administratorService.addAdministrator(a);
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            KeycloakApi.setRoleToUser(a.getUserName(), "MyINPulse-admin", token);
 | 
				
			||||||
 | 
					        } catch (Exception err) {
 | 
				
			||||||
 | 
					            logger.error(err);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void validateEntrepreneurAccount(long userId, String token) {
 | 
				
			||||||
 | 
					        Entrepreneur e = this.entrepreneurService.getEntrepreneurById(userId);
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            KeycloakApi.setRoleToUser(e.getUserName(), "MyINPulse-entrepreneur", token);
 | 
				
			||||||
 | 
					        } catch (Exception err) {
 | 
				
			||||||
 | 
					            logger.error(err);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        this.entrepreneurService.validateEntrepreneurById(userId);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<User> getPendingUsers() {
 | 
				
			||||||
 | 
					        return this.userService.getPendingAccounts();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void createAccount(
 | 
				
			||||||
 | 
					            String username,
 | 
				
			||||||
 | 
					            String userSurname,
 | 
				
			||||||
 | 
					            String primaryMail,
 | 
				
			||||||
 | 
					            String secondaryMail,
 | 
				
			||||||
 | 
					            String phoneNumber) {
 | 
				
			||||||
 | 
					        Administrator a =
 | 
				
			||||||
 | 
					                new Administrator(username, userSurname, primaryMail, secondaryMail, phoneNumber);
 | 
				
			||||||
 | 
					        this.administratorService.addAdministrator(a);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<Administrator> getAllAdmins() {
 | 
				
			||||||
 | 
					        return this.administratorService.allAdministrators();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,9 +1,13 @@
 | 
				
			|||||||
package enseirb.myinpulse.service;
 | 
					package enseirb.myinpulse.service;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.PENDING;
 | 
				
			||||||
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.ACTIVE;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Entrepreneur;
 | 
				
			||||||
import enseirb.myinpulse.model.Project;
 | 
					import enseirb.myinpulse.model.Project;
 | 
				
			||||||
import enseirb.myinpulse.model.SectionCell;
 | 
					import enseirb.myinpulse.model.SectionCell;
 | 
				
			||||||
import enseirb.myinpulse.service.database.ProjectService;
 | 
					import enseirb.myinpulse.model.User;
 | 
				
			||||||
import enseirb.myinpulse.service.database.SectionCellService;
 | 
					import enseirb.myinpulse.service.database.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.apache.logging.log4j.LogManager;
 | 
					import org.apache.logging.log4j.LogManager;
 | 
				
			||||||
import org.apache.logging.log4j.Logger;
 | 
					import org.apache.logging.log4j.Logger;
 | 
				
			||||||
@@ -12,6 +16,10 @@ import org.springframework.http.HttpStatus;
 | 
				
			|||||||
import org.springframework.stereotype.Service;
 | 
					import org.springframework.stereotype.Service;
 | 
				
			||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					import org.springframework.web.server.ResponseStatusException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.time.LocalDateTime;
 | 
				
			||||||
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Service
 | 
					@Service
 | 
				
			||||||
public class EntrepreneurApiService {
 | 
					public class EntrepreneurApiService {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -20,24 +28,39 @@ public class EntrepreneurApiService {
 | 
				
			|||||||
    private final SectionCellService sectionCellService;
 | 
					    private final SectionCellService sectionCellService;
 | 
				
			||||||
    private final ProjectService projectService;
 | 
					    private final ProjectService projectService;
 | 
				
			||||||
    private final UtilsService utilsService;
 | 
					    private final UtilsService utilsService;
 | 
				
			||||||
 | 
					    private final UserService userService;
 | 
				
			||||||
 | 
					    private final EntrepreneurService entrepreneurService;
 | 
				
			||||||
 | 
					    private final AdministratorService administratorService;
 | 
				
			||||||
 | 
					    private final AppointmentService appointmentService;
 | 
				
			||||||
 | 
					    private final AnnotationService annotationService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Autowired
 | 
					    @Autowired
 | 
				
			||||||
    EntrepreneurApiService(
 | 
					    EntrepreneurApiService(
 | 
				
			||||||
            SectionCellService sectionCellService,
 | 
					            SectionCellService sectionCellService,
 | 
				
			||||||
            ProjectService projectService,
 | 
					            ProjectService projectService,
 | 
				
			||||||
            UtilsService utilsService) {
 | 
					            UtilsService utilsService,
 | 
				
			||||||
 | 
					            UserService userService,
 | 
				
			||||||
 | 
					            EntrepreneurService entrepreneurService,
 | 
				
			||||||
 | 
					            AdministratorService administratorService,
 | 
				
			||||||
 | 
					            AppointmentService appointmentService,
 | 
				
			||||||
 | 
					            AnnotationService annotationService) {
 | 
				
			||||||
        this.sectionCellService = sectionCellService;
 | 
					        this.sectionCellService = sectionCellService;
 | 
				
			||||||
        this.projectService = projectService;
 | 
					        this.projectService = projectService;
 | 
				
			||||||
        this.utilsService = utilsService;
 | 
					        this.utilsService = utilsService;
 | 
				
			||||||
 | 
					        this.userService = userService;
 | 
				
			||||||
 | 
					        this.entrepreneurService = entrepreneurService;
 | 
				
			||||||
 | 
					        this.administratorService = administratorService;
 | 
				
			||||||
 | 
					        this.appointmentService = appointmentService;
 | 
				
			||||||
 | 
					        this.annotationService = annotationService;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void editSectionCell(Long sectionCellId, SectionCell sectionCell, String mail) {
 | 
					    public void editSectionCell(Long sectionCellId, String content, String mail) {
 | 
				
			||||||
        SectionCell editSectionCell = sectionCellService.getSectionCellById(sectionCellId);
 | 
					        if (sectionCellId == null) {
 | 
				
			||||||
        if (editSectionCell == null) {
 | 
					            logger.warn("Trying to edit unknown section cell");
 | 
				
			||||||
            System.err.println("Trying to edit unknown section cell");
 | 
					 | 
				
			||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas");
 | 
					                    HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        SectionCell sectionCell = sectionCellService.getSectionCellById(sectionCellId);
 | 
				
			||||||
        if (!utilsService.isAllowedToCheckProject(
 | 
					        if (!utilsService.isAllowedToCheckProject(
 | 
				
			||||||
                mail, this.sectionCellService.getProjectId(sectionCellId))) {
 | 
					                mail, this.sectionCellService.getProjectId(sectionCellId))) {
 | 
				
			||||||
            logger.warn(
 | 
					            logger.warn(
 | 
				
			||||||
@@ -53,27 +76,45 @@ public class EntrepreneurApiService {
 | 
				
			|||||||
                mail,
 | 
					                mail,
 | 
				
			||||||
                sectionCellId,
 | 
					                sectionCellId,
 | 
				
			||||||
                this.sectionCellService.getProjectId(sectionCellId));
 | 
					                this.sectionCellService.getProjectId(sectionCellId));
 | 
				
			||||||
        sectionCellService.updateSectionCell(
 | 
					        SectionCell newSectionCell =
 | 
				
			||||||
                sectionCellId,
 | 
					                new SectionCell(
 | 
				
			||||||
                sectionCell.getSectionId(),
 | 
					                        null,
 | 
				
			||||||
                sectionCell.getContentSectionCell(),
 | 
					                        sectionCell.getSectionId(),
 | 
				
			||||||
                sectionCell.getModificationDate());
 | 
					                        content,
 | 
				
			||||||
 | 
					                        LocalDateTime.now(),
 | 
				
			||||||
 | 
					                        sectionCell.getProjectSectionCell());
 | 
				
			||||||
 | 
					        newSectionCell.setIdReference(sectionCell.getIdReference());
 | 
				
			||||||
 | 
					        this.addSectionCell(newSectionCell, mail);
 | 
				
			||||||
 | 
					        sectionCell
 | 
				
			||||||
 | 
					                .getAppointmentSectionCell()
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        appointment -> {
 | 
				
			||||||
 | 
					                            this.appointmentService.updateAppointmentListSectionCell(
 | 
				
			||||||
 | 
					                                    appointment.getIdAppointment(), newSectionCell);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					        sectionCell
 | 
				
			||||||
 | 
					                .getListAnnotation()
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        annotation -> {
 | 
				
			||||||
 | 
					                            this.annotationService.updateAnnotationSectionCell(
 | 
				
			||||||
 | 
					                                    annotation.getIdAnnotation(), newSectionCell);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void removeSectionCell(Long sectionCellId, String mail) {
 | 
					    public void removeSectionCell(Long sectionCellId, String mail) {
 | 
				
			||||||
        SectionCell editSectionCell = sectionCellService.getSectionCellById(sectionCellId);
 | 
					        if (sectionCellId == null) {
 | 
				
			||||||
        if (editSectionCell == null) {
 | 
					            logger.warn("Trying to remove unknown section cell");
 | 
				
			||||||
            System.err.println("Trying to remove unknown section cell");
 | 
					 | 
				
			||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas");
 | 
					                    HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        SectionCell editSectionCell = sectionCellService.getSectionCellById(sectionCellId);
 | 
				
			||||||
        if (!utilsService.isAllowedToCheckProject(
 | 
					        if (!utilsService.isAllowedToCheckProject(
 | 
				
			||||||
                mail, this.sectionCellService.getProjectId(sectionCellId))) {
 | 
					                mail, this.sectionCellService.getProjectId(sectionCellId))) {
 | 
				
			||||||
            logger.warn(
 | 
					            logger.warn(
 | 
				
			||||||
                    "User {} tried to remove section cells {} of the project {} but is not allowed to.",
 | 
					                    "User {} tried to remove section cells {} of the project {} but is not allowed to.",
 | 
				
			||||||
                    mail,
 | 
					                    mail,
 | 
				
			||||||
                    sectionCellId,
 | 
					                    sectionCellId,
 | 
				
			||||||
                    this.sectionCellService.getSectionCellById(sectionCellId));
 | 
					                    this.sectionCellService.getProjectId(sectionCellId));
 | 
				
			||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
					                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -82,39 +123,162 @@ public class EntrepreneurApiService {
 | 
				
			|||||||
                mail,
 | 
					                mail,
 | 
				
			||||||
                sectionCellId,
 | 
					                sectionCellId,
 | 
				
			||||||
                this.sectionCellService.getProjectId(sectionCellId));
 | 
					                this.sectionCellService.getProjectId(sectionCellId));
 | 
				
			||||||
        sectionCellService.removeSectionCellById(sectionCellId);
 | 
					        SectionCell removedSectionCell =
 | 
				
			||||||
 | 
					                new SectionCell(
 | 
				
			||||||
 | 
					                        null,
 | 
				
			||||||
 | 
					                        -1L,
 | 
				
			||||||
 | 
					                        "",
 | 
				
			||||||
 | 
					                        LocalDateTime.now(),
 | 
				
			||||||
 | 
					                        this.projectService.getProjectById(
 | 
				
			||||||
 | 
					                                editSectionCell.getProjectSectionCell().getIdProject()));
 | 
				
			||||||
 | 
					        sectionCellService.addNewSectionCell(removedSectionCell);
 | 
				
			||||||
 | 
					        this.sectionCellService.updateSectionCellReferenceId(
 | 
				
			||||||
 | 
					                removedSectionCell.getIdSectionCell(), editSectionCell.getIdReference());
 | 
				
			||||||
 | 
					        projectService.updateProjectListSectionCell(
 | 
				
			||||||
 | 
					                sectionCellService.getProjectId(sectionCellId), removedSectionCell);
 | 
				
			||||||
 | 
					        // sectionCellService.removeSectionCellById(sectionCellId);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void addSectionCell(SectionCell sectionCell, String mail) {
 | 
					    public void addSectionCell(SectionCell sectionCell, String mail) {
 | 
				
			||||||
        if (sectionCell == null) {
 | 
					        if (sectionCell == null) {
 | 
				
			||||||
            System.err.println("Trying to create an empty section cell");
 | 
					            logger.warn("Trying to create an empty section cell");
 | 
				
			||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.BAD_REQUEST, "La cellule de section fournie est vide");
 | 
					                    HttpStatus.BAD_REQUEST, "La cellule de section fournie est vide");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (sectionCell.getSectionId() == -1) {
 | 
				
			||||||
 | 
					            logger.warn("Trying to create an illegal section cell");
 | 
				
			||||||
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
 | 
					                    HttpStatus.BAD_REQUEST, "La cellule de section fournie n'est pas valide");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        if (!utilsService.isAllowedToCheckProject(
 | 
					        if (!utilsService.isAllowedToCheckProject(
 | 
				
			||||||
                mail, this.sectionCellService.getProjectId(sectionCell.getIdSectionCell()))) {
 | 
					                mail, sectionCell.getProjectSectionCell().getIdProject())) {
 | 
				
			||||||
            logger.warn(
 | 
					            logger.warn(
 | 
				
			||||||
                    "User {} tried to add a section cell to the project {} but is not allowed to.",
 | 
					                    "User {} tried to add a section cell to the project {} but is not allowed to.",
 | 
				
			||||||
                    mail,
 | 
					                    mail,
 | 
				
			||||||
                    this.sectionCellService.getProjectId(sectionCell.getIdSectionCell()));
 | 
					                    sectionCell.getProjectSectionCell().getIdProject());
 | 
				
			||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
					                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        logger.info(
 | 
					        logger.info(
 | 
				
			||||||
                "User {} added a new section cell {} to the project with id {}",
 | 
					                "User {} added a new section cell {} to the project {}",
 | 
				
			||||||
                mail,
 | 
					                mail,
 | 
				
			||||||
                sectionCell.getIdSectionCell(),
 | 
					                sectionCell.getIdSectionCell(),
 | 
				
			||||||
                this.sectionCellService.getProjectId(sectionCell.getIdSectionCell()));
 | 
					                sectionCell.getProjectSectionCell().getIdProject());
 | 
				
			||||||
        sectionCellService.addNewSectionCell(sectionCell);
 | 
					        SectionCell newSectionCell =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        sectionCell); // if here, logger fails cause id is null (not added yet)
 | 
				
			||||||
 | 
					        newSectionCell.getProjectSectionCell().updateListSectionCell(newSectionCell);
 | 
				
			||||||
 | 
					        newSectionCell
 | 
				
			||||||
 | 
					                .getAppointmentSectionCell()
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        appointment -> {
 | 
				
			||||||
 | 
					                            this.appointmentService.updateAppointmentListSectionCell(
 | 
				
			||||||
 | 
					                                    appointment.getIdAppointment(), newSectionCell);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					        newSectionCell
 | 
				
			||||||
 | 
					                .getListAnnotation()
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        annotation -> {
 | 
				
			||||||
 | 
					                            this.annotationService.updateAnnotationSectionCell(
 | 
				
			||||||
 | 
					                                    annotation.getIdAnnotation(), newSectionCell);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void requestNewProject(Project project, String mail) {
 | 
					    public void requestNewProject(Project project, String mail) {
 | 
				
			||||||
        if (project == null) {
 | 
					        if (project == null) {
 | 
				
			||||||
            logger.error("Trying to request the creation of a null project");
 | 
					            logger.warn("Trying to request the creation of a null project");
 | 
				
			||||||
            throw new ResponseStatusException(HttpStatus.BAD_REQUEST, "Le projet fourni est vide");
 | 
					            throw new ResponseStatusException(HttpStatus.BAD_REQUEST, "Le projet fourni est vide");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        logger.info("User {} created a new project with id {}", mail, project.getIdProject());
 | 
					        logger.info("User {} created a new project named {}", mail, project.getProjectName());
 | 
				
			||||||
        project.setProjectStatus("PENDING");
 | 
					        project.setEntrepreneurProposed((Entrepreneur) this.userService.getUserByEmail(mail));
 | 
				
			||||||
        projectService.addNewProject(project);
 | 
					        projectService.addNewProject(project);
 | 
				
			||||||
 | 
					        this.projectService.updateProjectStatus(project.getIdProject(), PENDING);
 | 
				
			||||||
 | 
					        if (project.getProjectAdministrator() != null) {
 | 
				
			||||||
 | 
					            this.administratorService.updateAdministratorListProject(
 | 
				
			||||||
 | 
					                    project.getProjectAdministrator().getIdUser(), project);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        this.entrepreneurService.updateEntrepreneurProjectProposed(
 | 
				
			||||||
 | 
					                this.userService.getUserByEmail(mail).getIdUser(), project);
 | 
				
			||||||
 | 
					        this.entrepreneurService.updateEntrepreneurProjectParticipation(
 | 
				
			||||||
 | 
					                this.userService.getUserByEmail(mail).getIdUser(), project);
 | 
				
			||||||
 | 
					        project.getListEntrepreneurParticipation()
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        entrepreneur ->
 | 
				
			||||||
 | 
					                                this.entrepreneurService.updateEntrepreneurProjectParticipation(
 | 
				
			||||||
 | 
					                                        entrepreneur.getIdUser(), project));
 | 
				
			||||||
 | 
					        project.getListSectionCell()
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        sectionCell ->
 | 
				
			||||||
 | 
					                                this.sectionCellService.updateSectionCellProject(
 | 
				
			||||||
 | 
					                                        sectionCell.getIdSectionCell(), project));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void createAccount(Entrepreneur e) {
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            userService.getUserByEmail(e.getPrimaryMail());
 | 
				
			||||||
 | 
					            logger.error("The user {} already exists in the system", e.getPrimaryMail());
 | 
				
			||||||
 | 
					        } catch (ResponseStatusException err) {
 | 
				
			||||||
 | 
					            this.entrepreneurService.addEntrepreneur(e);
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        throw new ResponseStatusException(HttpStatus.CONFLICT, "User already exists in the system");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<Project> getProjectIdViaClaim(String email) {
 | 
				
			||||||
 | 
					        Long UserId = this.userService.getUserByEmail(email).getIdUser();
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = this.entrepreneurService.getEntrepreneurById(UserId);
 | 
				
			||||||
 | 
					        List<Project> Project_List = new ArrayList<>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Project_List.add(entrepreneur.getProjectParticipation());
 | 
				
			||||||
 | 
					        return Project_List;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<Entrepreneur> getAllEntrepreneurs() {
 | 
				
			||||||
 | 
					        return entrepreneurService.getAllEntrepreneurs();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Checks if an entrepreneur with the given email has a project that is ACTIVE.
 | 
				
			||||||
 | 
					     *
 | 
				
			||||||
 | 
					     * @param email The email of the entrepreneur.
 | 
				
			||||||
 | 
					     * @return true if the entrepreneur has an active project, false otherwise.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    public Boolean checkIfEntrepreneurProjectActive(String email) {
 | 
				
			||||||
 | 
					        User user = this.userService.getUserByEmail(email);
 | 
				
			||||||
 | 
					        if (user == null) {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Long userId = user.getIdUser();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = this.entrepreneurService.getEntrepreneurById(userId);
 | 
				
			||||||
 | 
					        if (entrepreneur == null) {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Project proposedProject = entrepreneur.getProjectProposed();
 | 
				
			||||||
 | 
					        return proposedProject != null && proposedProject.getProjectStatus() == ACTIVE;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Checks if an entrepreneur with the given email has proposed a project.
 | 
				
			||||||
 | 
					     *
 | 
				
			||||||
 | 
					     * @param email The email of the entrepreneur.
 | 
				
			||||||
 | 
					     * @return true if the entrepreneur has a proposed project, false otherwise.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    public Boolean entrepreneurHasPendingRequestedProject(String email) {
 | 
				
			||||||
 | 
					        User user = this.userService.getUserByEmail(email);
 | 
				
			||||||
 | 
					        if (user == null) {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Long userId = user.getIdUser();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = this.entrepreneurService.getEntrepreneurById(userId);
 | 
				
			||||||
 | 
					        if (entrepreneur == null) {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Project proposedProject = entrepreneur.getProjectProposed();
 | 
				
			||||||
 | 
					        if (entrepreneur.getProjectProposed() == null) {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return proposedProject.getProjectStatus() == PENDING;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,12 +6,17 @@ import enseirb.myinpulse.exception.UserNotFoundException;
 | 
				
			|||||||
import enseirb.myinpulse.model.RoleRepresentation;
 | 
					import enseirb.myinpulse.model.RoleRepresentation;
 | 
				
			||||||
import enseirb.myinpulse.model.UserRepresentation;
 | 
					import enseirb.myinpulse.model.UserRepresentation;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.apache.logging.log4j.LogManager;
 | 
				
			||||||
 | 
					import org.apache.logging.log4j.Logger;
 | 
				
			||||||
import org.springframework.web.client.RestClient;
 | 
					import org.springframework.web.client.RestClient;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import javax.management.relation.RoleNotFoundException;
 | 
					import javax.management.relation.RoleNotFoundException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
public class KeycloakApi {
 | 
					public class KeycloakApi {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    protected static final Logger logger = LogManager.getLogger();
 | 
				
			||||||
    static final String keycloakUrl;
 | 
					    static final String keycloakUrl;
 | 
				
			||||||
    static final String realmName;
 | 
					    static final String realmName;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -29,44 +34,48 @@ public class KeycloakApi {
 | 
				
			|||||||
        realmName = System.getenv("VITE_KEYCLOAK_REALM");
 | 
					        realmName = System.getenv("VITE_KEYCLOAK_REALM");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    static String toBearer(String b) {
 | 
				
			||||||
 | 
					        return "Bearer " + b;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * Uses Keycloak API to retrieve a role representation of a role by its name
 | 
					     * Uses Keycloak API to retrieve a role representation of a role by its name
 | 
				
			||||||
     *
 | 
					     *
 | 
				
			||||||
     * @param roleName name of the role
 | 
					     * @param roleName name of the role
 | 
				
			||||||
     * @param bearer authorization header used by the client to authenticate to keycloak
 | 
					     * @param token authorization header used by the client to authenticate to keycloak
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public static RoleRepresentation getRoleRepresentationByName(String roleName, String bearer)
 | 
					    public static RoleRepresentation getRoleRepresentationByName(String roleName, String token)
 | 
				
			||||||
            throws RoleNotFoundException {
 | 
					            throws RoleNotFoundException {
 | 
				
			||||||
        RoleRepresentation[] response =
 | 
					        RoleRepresentation response =
 | 
				
			||||||
                RestClient.builder()
 | 
					                RestClient.builder()
 | 
				
			||||||
                        .baseUrl(keycloakUrl)
 | 
					                        .baseUrl(keycloakUrl)
 | 
				
			||||||
                        .defaultHeader("Authorization", bearer)
 | 
					                        .defaultHeader("Authorization", toBearer(token))
 | 
				
			||||||
                        .build()
 | 
					                        .build()
 | 
				
			||||||
                        .get()
 | 
					                        .get()
 | 
				
			||||||
                        .uri("/admin/realms/{realmName}/roles/{roleName}", realmName, roleName)
 | 
					                        .uri("/admin/realms/{realmName}/roles/{roleName}", realmName, roleName)
 | 
				
			||||||
                        .retrieve()
 | 
					                        .retrieve()
 | 
				
			||||||
                        .body(RoleRepresentation[].class);
 | 
					                        .body(RoleRepresentation.class);
 | 
				
			||||||
 | 
					        /*
 | 
				
			||||||
        if (response == null || response.length == 0) {
 | 
					        {"id":"7a845f2e-c832-4465-8cd8-894d72bc13f1","name":"MyINPulse-entrepreneur","description":"Role for entrepreneur","composite":false,"clientRole":false,"containerId":"0d6f691b-e328-471a-b89e-c30bd7e5b6b0","attributes":{}}
 | 
				
			||||||
            throw new RoleNotFoundException("Role not found");
 | 
					         */
 | 
				
			||||||
        }
 | 
					        // TODO: check what happens when role does not exist
 | 
				
			||||||
        return response[0];
 | 
					        return response;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * Use keycloak API to to retreive a userID via his name or email.
 | 
					     * Use keycloak API to to retreive a userID via his name or email.
 | 
				
			||||||
     *
 | 
					     *
 | 
				
			||||||
     * @param username username or mail of the user
 | 
					     * @param username username or mail of the user
 | 
				
			||||||
     * @param bearer bearer of the user, allowing access to database
 | 
					     * @param token bearer of the user, allowing access to database
 | 
				
			||||||
     * @return the userid, as a String
 | 
					     * @return the userid, as a String
 | 
				
			||||||
     * @throws UserNotFoundException
 | 
					     * @throws UserNotFoundException
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public static String getUserIdByName(String username, String bearer)
 | 
					    public static String getUserIdByName(String username, String token)
 | 
				
			||||||
            throws UserNotFoundException {
 | 
					            throws UserNotFoundException {
 | 
				
			||||||
        UserRepresentation[] response =
 | 
					        UserRepresentation[] response =
 | 
				
			||||||
                RestClient.builder()
 | 
					                RestClient.builder()
 | 
				
			||||||
                        .baseUrl(keycloakUrl)
 | 
					                        .baseUrl(keycloakUrl)
 | 
				
			||||||
                        .defaultHeader("Authorization", bearer)
 | 
					                        .defaultHeader("Authorization", toBearer(token))
 | 
				
			||||||
                        .build()
 | 
					                        .build()
 | 
				
			||||||
                        .get()
 | 
					                        .get()
 | 
				
			||||||
                        .uri(
 | 
					                        .uri(
 | 
				
			||||||
@@ -91,27 +100,26 @@ public class KeycloakApi {
 | 
				
			|||||||
     *
 | 
					     *
 | 
				
			||||||
     * @param username
 | 
					     * @param username
 | 
				
			||||||
     * @param roleName
 | 
					     * @param roleName
 | 
				
			||||||
     * @param bearer
 | 
					     * @param token
 | 
				
			||||||
     * @throws RoleNotFoundException
 | 
					     * @throws RoleNotFoundException
 | 
				
			||||||
     * @throws UserNotFoundException
 | 
					     * @throws UserNotFoundException
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public static void setRoleToUser(String username, String roleName, String bearer)
 | 
					    public static void setRoleToUser(String username, String roleName, String token)
 | 
				
			||||||
            throws RoleNotFoundException, UserNotFoundException {
 | 
					            throws RoleNotFoundException, UserNotFoundException {
 | 
				
			||||||
        RoleRepresentation roleRepresentation = getRoleRepresentationByName(roleName, bearer);
 | 
					        RoleRepresentation roleRepresentation = getRoleRepresentationByName(roleName, token);
 | 
				
			||||||
        String userId = getUserIdByName(username, bearer);
 | 
					        String userId = getUserIdByName(username, token);
 | 
				
			||||||
 | 
					        List<RoleRepresentation> rolesToAdd = List.of(roleRepresentation);
 | 
				
			||||||
 | 
					        logger.debug("Adding role {} to user {}", roleRepresentation.id, userId);
 | 
				
			||||||
        RestClient.builder()
 | 
					        RestClient.builder()
 | 
				
			||||||
                .baseUrl(keycloakUrl)
 | 
					                .baseUrl(keycloakUrl)
 | 
				
			||||||
                .defaultHeader("Authorization", bearer)
 | 
					                .defaultHeader("Authorization", toBearer(token))
 | 
				
			||||||
                .build()
 | 
					                .build()
 | 
				
			||||||
                .post()
 | 
					                .post()
 | 
				
			||||||
                .uri(
 | 
					                .uri("/admin/realms/" + realmName + "/users/" + userId + "/role-mappings/realm")
 | 
				
			||||||
                        "/admin/realms/${realmName}/users/${userId}/role-mappings/realm",
 | 
					                .body(rolesToAdd)
 | 
				
			||||||
                        realmName,
 | 
					 | 
				
			||||||
                        userId)
 | 
					 | 
				
			||||||
                .body(roleRepresentation)
 | 
					 | 
				
			||||||
                .contentType(APPLICATION_JSON)
 | 
					                .contentType(APPLICATION_JSON)
 | 
				
			||||||
                .retrieve();
 | 
					                .retrieve()
 | 
				
			||||||
 | 
					                .toBodilessEntity();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,9 @@
 | 
				
			|||||||
package enseirb.myinpulse.service;
 | 
					package enseirb.myinpulse.service;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.itextpdf.text.*;
 | 
				
			||||||
 | 
					import com.itextpdf.text.pdf.PdfWriter;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.controller.AdminApi;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.controller.EntrepreneurApi;
 | 
				
			||||||
import enseirb.myinpulse.model.*;
 | 
					import enseirb.myinpulse.model.*;
 | 
				
			||||||
import enseirb.myinpulse.service.database.*;
 | 
					import enseirb.myinpulse.service.database.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -10,14 +14,30 @@ import org.springframework.http.HttpStatus;
 | 
				
			|||||||
import org.springframework.stereotype.Service;
 | 
					import org.springframework.stereotype.Service;
 | 
				
			||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					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.LocalDateTime;
 | 
				
			||||||
import java.time.format.DateTimeFormatter;
 | 
					import java.time.format.DateTimeFormatter;
 | 
				
			||||||
import java.util.ArrayList;
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.HashMap;
 | 
				
			||||||
 | 
					import java.util.HashSet;
 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					import java.util.Map;
 | 
				
			||||||
 | 
					import java.util.Set;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Service
 | 
					@Service
 | 
				
			||||||
public class SharedApiService {
 | 
					public class SharedApiService {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private final AdminApi adminApi;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private final EntrepreneurApi entrepreneurApi;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    protected static final Logger logger = LogManager.getLogger();
 | 
					    protected static final Logger logger = LogManager.getLogger();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    private final ProjectService projectService;
 | 
					    private final ProjectService projectService;
 | 
				
			||||||
@@ -33,12 +53,16 @@ public class SharedApiService {
 | 
				
			|||||||
            EntrepreneurService entrepreneurService,
 | 
					            EntrepreneurService entrepreneurService,
 | 
				
			||||||
            SectionCellService sectionCellService,
 | 
					            SectionCellService sectionCellService,
 | 
				
			||||||
            AppointmentService appointmentService,
 | 
					            AppointmentService appointmentService,
 | 
				
			||||||
            UtilsService utilsService) {
 | 
					            UtilsService utilsService,
 | 
				
			||||||
 | 
					            EntrepreneurApi entrepreneurApi,
 | 
				
			||||||
 | 
					            AdminApi adminApi) {
 | 
				
			||||||
        this.projectService = projectService;
 | 
					        this.projectService = projectService;
 | 
				
			||||||
        this.entrepreneurService = entrepreneurService;
 | 
					        this.entrepreneurService = entrepreneurService;
 | 
				
			||||||
        this.sectionCellService = sectionCellService;
 | 
					        this.sectionCellService = sectionCellService;
 | 
				
			||||||
        this.appointmentService = appointmentService;
 | 
					        this.appointmentService = appointmentService;
 | 
				
			||||||
        this.utilsService = utilsService;
 | 
					        this.utilsService = utilsService;
 | 
				
			||||||
 | 
					        this.entrepreneurApi = entrepreneurApi;
 | 
				
			||||||
 | 
					        this.adminApi = adminApi;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO filter this with date
 | 
					    // TODO filter this with date
 | 
				
			||||||
@@ -58,10 +82,54 @@ public class SharedApiService {
 | 
				
			|||||||
        LocalDateTime dateTime = LocalDateTime.parse(date, formatter);
 | 
					        LocalDateTime dateTime = LocalDateTime.parse(date, formatter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        Project project = this.projectService.getProjectById(projectId);
 | 
					        Project project = this.projectService.getProjectById(projectId);
 | 
				
			||||||
        return this.sectionCellService.getSectionCellsByProjectAndSectionIdBeforeDate(
 | 
					        return this.sectionCellService.getLatestSectionCellsByIdReferenceBeforeDate(
 | 
				
			||||||
                project, sectionId, dateTime);
 | 
					                project, sectionId, dateTime);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Retrieve all up to date (for every sectionId) sectionCells of a project
 | 
				
			||||||
 | 
					    public Iterable<SectionCell> getAllSectionCells(long projectId, 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");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Project project = this.projectService.getProjectById(projectId);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Map<Long, SectionCell> latestSectionCellsMap =
 | 
				
			||||||
 | 
					                new HashMap<>(); // List for the intermediate result
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Iterate through all SectionCells associated with the project
 | 
				
			||||||
 | 
					        // This loop iterates over project.getListSectionCell() but does NOT modify it which causes
 | 
				
			||||||
 | 
					        // ConcurrentModificationException.
 | 
				
			||||||
 | 
					        // Modifications are done only on the latestSectionCellsMap (which is safe).
 | 
				
			||||||
 | 
					        project.getListSectionCell() // <-- Iterating over the original list (read-only)
 | 
				
			||||||
 | 
					                .forEach(
 | 
				
			||||||
 | 
					                        projectCell -> {
 | 
				
			||||||
 | 
					                            Long idReference = projectCell.getIdReference();
 | 
				
			||||||
 | 
					                            // Check if we have already seen a SectionCell with this idReference in
 | 
				
			||||||
 | 
					                            // our map
 | 
				
			||||||
 | 
					                            if (latestSectionCellsMap.containsKey(idReference)) {
 | 
				
			||||||
 | 
					                                SectionCell existingCell = latestSectionCellsMap.get(idReference);
 | 
				
			||||||
 | 
					                                // Compare modification dates. If the current cell is newer, replace
 | 
				
			||||||
 | 
					                                // the one in the map.
 | 
				
			||||||
 | 
					                                if (projectCell
 | 
				
			||||||
 | 
					                                        .getModificationDate()
 | 
				
			||||||
 | 
					                                        .isAfter(existingCell.getModificationDate())) {
 | 
				
			||||||
 | 
					                                    latestSectionCellsMap.put(idReference, projectCell);
 | 
				
			||||||
 | 
					                                }
 | 
				
			||||||
 | 
					                            } else {
 | 
				
			||||||
 | 
					                                // If this is the first time we encounter this idReference, add the
 | 
				
			||||||
 | 
					                                // cell to the map.
 | 
				
			||||||
 | 
					                                latestSectionCellsMap.put(idReference, projectCell);
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					        return new ArrayList<>(latestSectionCellsMap.values());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO: test
 | 
					    // TODO: test
 | 
				
			||||||
    public Iterable<Entrepreneur> getEntrepreneursByProjectId(long projectId, String mail) {
 | 
					    public Iterable<Entrepreneur> getEntrepreneursByProjectId(long projectId, String mail) {
 | 
				
			||||||
        if (!utilsService.isAllowedToCheckProject(mail, projectId)) {
 | 
					        if (!utilsService.isAllowedToCheckProject(mail, projectId)) {
 | 
				
			||||||
@@ -87,10 +155,9 @@ public class SharedApiService {
 | 
				
			|||||||
                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
					                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        Project project = this.projectService.getProjectById(projectId);
 | 
					        Project project = this.projectService.getProjectById(projectId);
 | 
				
			||||||
        return project.getAdministrator();
 | 
					        return project.getProjectAdministrator();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO
 | 
					 | 
				
			||||||
    public Iterable<Appointment> getAppointmentsByProjectId(long projectId, String mail) {
 | 
					    public Iterable<Appointment> getAppointmentsByProjectId(long projectId, String mail) {
 | 
				
			||||||
        if (!utilsService.isAllowedToCheckProject(mail, projectId)) {
 | 
					        if (!utilsService.isAllowedToCheckProject(mail, projectId)) {
 | 
				
			||||||
            logger.warn(
 | 
					            logger.warn(
 | 
				
			||||||
@@ -104,22 +171,30 @@ public class SharedApiService {
 | 
				
			|||||||
                "User {} tried to check the appointments related to the project {}",
 | 
					                "User {} tried to check the appointments related to the project {}",
 | 
				
			||||||
                mail,
 | 
					                mail,
 | 
				
			||||||
                projectId);
 | 
					                projectId);
 | 
				
			||||||
        Iterable<SectionCell> sectionCells =
 | 
					
 | 
				
			||||||
                this.sectionCellService.getSectionCellsByProject(
 | 
					        Project project = projectService.getProjectById(projectId);
 | 
				
			||||||
                        projectService.getProjectById(projectId),
 | 
					
 | 
				
			||||||
                        2L); // sectionId useless in this function ?
 | 
					        Iterable<SectionCell> sectionCellsIterable =
 | 
				
			||||||
        List<Appointment> appointments = new ArrayList<Appointment>();
 | 
					                this.sectionCellService.getSectionCellsByProject(project);
 | 
				
			||||||
        sectionCells.forEach(
 | 
					
 | 
				
			||||||
 | 
					        // Use a Set to collect unique appointments
 | 
				
			||||||
 | 
					        Set<Appointment> uniqueAppointments = new HashSet<>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        sectionCellsIterable.forEach(
 | 
				
			||||||
                sectionCell -> {
 | 
					                sectionCell -> {
 | 
				
			||||||
                    appointments.addAll(
 | 
					                    List<Appointment> sectionAppointments =
 | 
				
			||||||
                            this.sectionCellService.getAppointmentsBySectionCellId(
 | 
					                            this.sectionCellService.getAppointmentsBySectionCellId(
 | 
				
			||||||
                                    sectionCell.getIdSectionCell()));
 | 
					                                    sectionCell.getIdSectionCell());
 | 
				
			||||||
 | 
					                    // Add all appointments from this section cell to the Set
 | 
				
			||||||
 | 
					                    uniqueAppointments.addAll(sectionAppointments);
 | 
				
			||||||
                });
 | 
					                });
 | 
				
			||||||
        return appointments;
 | 
					
 | 
				
			||||||
 | 
					        // Convert the Set back to a List for the return value
 | 
				
			||||||
 | 
					        return new ArrayList<>(uniqueAppointments);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO
 | 
					    public void getPDFReport(long appointmentId, String mail)
 | 
				
			||||||
    public void getPDFReport(long appointmentId, String mail) {
 | 
					            throws DocumentException, URISyntaxException, IOException {
 | 
				
			||||||
        long projectId =
 | 
					        long projectId =
 | 
				
			||||||
                this.appointmentService
 | 
					                this.appointmentService
 | 
				
			||||||
                        .getAppointmentById(appointmentId)
 | 
					                        .getAppointmentById(appointmentId)
 | 
				
			||||||
@@ -139,15 +214,111 @@ public class SharedApiService {
 | 
				
			|||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
					                    HttpStatus.UNAUTHORIZED, "You're not allowed to check this project");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        /* return this.appointmentService
 | 
					        logger.info(
 | 
				
			||||||
        .getAppointmentById(appointmentId)
 | 
					                "User {} generated the PDF report related to appointment {}", mail, appointmentId);
 | 
				
			||||||
        .getAppointmentReport().getReportContent(); */
 | 
					
 | 
				
			||||||
        // generate pdf from this string, and format it to be decent looking
 | 
					        String reportContent =
 | 
				
			||||||
        throw new ResponseStatusException(HttpStatus.NOT_IMPLEMENTED, "Not implemented yet");
 | 
					                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());
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO
 | 
					 | 
				
			||||||
    public void createAppointmentRequest(Appointment appointment, String mail) {
 | 
					    public void createAppointmentRequest(Appointment appointment, String mail) {
 | 
				
			||||||
        throw new ResponseStatusException(HttpStatus.NOT_IMPLEMENTED, "Not implemented yet");
 | 
					        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);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /*
 | 
				
			||||||
 | 
					         * On initial insertion, the resport value is null unless that report does already exist in the db somewhere
 | 
				
			||||||
 | 
					         * If a non null value is passed and it does not exist in db it will throw an exception
 | 
				
			||||||
 | 
					         */
 | 
				
			||||||
 | 
					        if (newAppointment.getAppointmentReport() != null) {
 | 
				
			||||||
 | 
					            newAppointment.getAppointmentReport().setAppointmentReport(newAppointment);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -15,6 +15,8 @@ import org.springframework.beans.factory.annotation.Autowired;
 | 
				
			|||||||
import org.springframework.stereotype.Service;
 | 
					import org.springframework.stereotype.Service;
 | 
				
			||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					import org.springframework.web.server.ResponseStatusException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.Objects;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Service
 | 
					@Service
 | 
				
			||||||
public class UtilsService {
 | 
					public class UtilsService {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -44,19 +46,36 @@ public class UtilsService {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
        User user = this.userService.getUserByEmail(mail);
 | 
					        User user = this.userService.getUserByEmail(mail);
 | 
				
			||||||
        Entrepreneur entrepreneur = this.entrepreneurService.getEntrepreneurById(user.getIdUser());
 | 
					        Entrepreneur entrepreneur = this.entrepreneurService.getEntrepreneurById(user.getIdUser());
 | 
				
			||||||
 | 
					        if (entrepreneur == null) {
 | 
				
			||||||
 | 
					            logger.debug("testing access with an unknown Entrepreneur");
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (entrepreneur.getProjectParticipation() == null) {
 | 
				
			||||||
 | 
					            logger.debug("testing access with an user with no project participation");
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        Project project = this.projectService.getProjectById(projectId);
 | 
					        Project project = this.projectService.getProjectById(projectId);
 | 
				
			||||||
        return entrepreneur.getProjectParticipation() == project;
 | 
					        // We compare the ID instead of the project themselves
 | 
				
			||||||
 | 
					        return Objects.equals(
 | 
				
			||||||
 | 
					                entrepreneur.getProjectParticipation().getIdProject(), project.getIdProject());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO: test
 | 
					    // TODO: test
 | 
				
			||||||
    Boolean isAnAdmin(String mail) {
 | 
					    public Boolean isAnAdmin(String mail) {
 | 
				
			||||||
        try {
 | 
					        try {
 | 
				
			||||||
            long userId = this.userService.getUserByEmail(mail).getIdUser();
 | 
					            long userId = this.userService.getUserByEmail(mail).getIdUser();
 | 
				
			||||||
            Administrator a = this.administratorService.getAdministratorById(userId);
 | 
					            Administrator a = this.administratorService.getAdministratorById(userId);
 | 
				
			||||||
            return true;
 | 
					            return true;
 | 
				
			||||||
        } catch (ResponseStatusException e) {
 | 
					        } catch (ResponseStatusException e) {
 | 
				
			||||||
            logger.info(e);
 | 
					            logger.info(e);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            return false;
 | 
					            return false;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Boolean checkEntrepreneurNotPending(String email) {
 | 
				
			||||||
 | 
					        // Throws 404 if user not found
 | 
				
			||||||
 | 
					        User user = userService.getUserByEmail(email);
 | 
				
			||||||
 | 
					        return !user.isPending();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,9 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.Administrator;
 | 
					import enseirb.myinpulse.model.Administrator;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Annotation;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.MakeAppointment;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Project;
 | 
				
			||||||
import enseirb.myinpulse.repository.AdministratorRepository;
 | 
					import enseirb.myinpulse.repository.AdministratorRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.apache.logging.log4j.LogManager;
 | 
					import org.apache.logging.log4j.LogManager;
 | 
				
			||||||
@@ -37,10 +40,64 @@ public class AdministratorService {
 | 
				
			|||||||
        return administrator.get();
 | 
					        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) {
 | 
					    public Administrator addAdministrator(Administrator administrator) {
 | 
				
			||||||
        return this.administratorRepository.save(administrator);
 | 
					        return this.administratorRepository.save(administrator);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAdministratorListProject(long idAdministrator, Project project) {
 | 
				
			||||||
 | 
					        Administrator administrator = getAdministratorById(idAdministrator);
 | 
				
			||||||
 | 
					        administrator.updateListProject(project);
 | 
				
			||||||
 | 
					        this.administratorRepository.save(administrator);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAdministratorListAnnotation(long idAdministrator, Annotation annotation) {
 | 
				
			||||||
 | 
					        Administrator administrator = getAdministratorById(idAdministrator);
 | 
				
			||||||
 | 
					        administrator.updateListAnnotation(annotation);
 | 
				
			||||||
 | 
					        this.administratorRepository.save(administrator);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAdministratorMakeAppointment(
 | 
				
			||||||
 | 
					            long idAdministrator, MakeAppointment makeAppointment) {
 | 
				
			||||||
 | 
					        Administrator administrator = getAdministratorById(idAdministrator);
 | 
				
			||||||
 | 
					        administrator.setMakeAppointment(makeAppointment);
 | 
				
			||||||
 | 
					        this.administratorRepository.save(administrator);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Administrator updateAdministrator(
 | 
				
			||||||
 | 
					            Long idAdministrator,
 | 
				
			||||||
 | 
					            Project project,
 | 
				
			||||||
 | 
					            Annotation annotation,
 | 
				
			||||||
 | 
					            MakeAppointment makeAppointment) {
 | 
				
			||||||
 | 
					        Optional<Administrator> administrator = administratorRepository.findById(idAdministrator);
 | 
				
			||||||
 | 
					        if (administrator.isEmpty()) {
 | 
				
			||||||
 | 
					            logger.error(
 | 
				
			||||||
 | 
					                    "updateAdministrator : No administrator found with id {}", idAdministrator);
 | 
				
			||||||
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
 | 
					                    HttpStatus.NOT_FOUND, "Cet administrateur n'existe pas");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (project != null) {
 | 
				
			||||||
 | 
					            administrator.get().updateListProject(project);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (annotation != null) {
 | 
				
			||||||
 | 
					            administrator.get().updateListAnnotation(annotation);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (makeAppointment != null) {
 | 
				
			||||||
 | 
					            administrator.get().setMakeAppointment(makeAppointment);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return this.administratorRepository.save(administrator.get());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /*
 | 
					    /*
 | 
				
			||||||
    public Administrator getAdministratorByProject(Project project) {
 | 
					    public Administrator getAdministratorByProject(Project project) {
 | 
				
			||||||
        r
 | 
					        r
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,8 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Administrator;
 | 
				
			||||||
import enseirb.myinpulse.model.Annotation;
 | 
					import enseirb.myinpulse.model.Annotation;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.SectionCell;
 | 
				
			||||||
import enseirb.myinpulse.repository.AnnotationRepository;
 | 
					import enseirb.myinpulse.repository.AnnotationRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.apache.logging.log4j.LogManager;
 | 
					import org.apache.logging.log4j.LogManager;
 | 
				
			||||||
@@ -46,6 +48,12 @@ public class AnnotationService {
 | 
				
			|||||||
        this.annotationRepository.deleteById(id);
 | 
					        this.annotationRepository.deleteById(id);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAnnotationComment(long idAnnotation, String comment) {
 | 
				
			||||||
 | 
					        Annotation annotation = getAnnotationById(idAnnotation);
 | 
				
			||||||
 | 
					        annotation.setComment(comment);
 | 
				
			||||||
 | 
					        this.annotationRepository.save(annotation);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Annotation updateAnnotation(Long id, String comment) {
 | 
					    public Annotation updateAnnotation(Long id, String comment) {
 | 
				
			||||||
        Optional<Annotation> annotation = annotationRepository.findById(id);
 | 
					        Optional<Annotation> annotation = annotationRepository.findById(id);
 | 
				
			||||||
        if (annotation.isEmpty()) {
 | 
					        if (annotation.isEmpty()) {
 | 
				
			||||||
@@ -58,4 +66,16 @@ public class AnnotationService {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
        return this.annotationRepository.save(annotation.get());
 | 
					        return this.annotationRepository.save(annotation.get());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAnnotationSectionCell(long idAnnotation, SectionCell sectionCell) {
 | 
				
			||||||
 | 
					        Annotation annotation = getAnnotationById(idAnnotation);
 | 
				
			||||||
 | 
					        annotation.setSectionCellAnnotation(sectionCell);
 | 
				
			||||||
 | 
					        this.annotationRepository.save(annotation);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAnnotationAdministrator(long idAnnotation, Administrator administrator) {
 | 
				
			||||||
 | 
					        Annotation annotation = getAnnotationById(idAnnotation);
 | 
				
			||||||
 | 
					        annotation.setAdministratorAnnotation(administrator);
 | 
				
			||||||
 | 
					        this.annotationRepository.save(annotation);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,7 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.Appointment;
 | 
					import enseirb.myinpulse.model.Appointment;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.SectionCell;
 | 
				
			||||||
import enseirb.myinpulse.repository.AppointmentRepository;
 | 
					import enseirb.myinpulse.repository.AppointmentRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.apache.logging.log4j.LogManager;
 | 
					import org.apache.logging.log4j.LogManager;
 | 
				
			||||||
@@ -47,13 +48,50 @@ public class AppointmentService {
 | 
				
			|||||||
        this.appointmentRepository.deleteById(id);
 | 
					        this.appointmentRepository.deleteById(id);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAppointmentDate(long idAppointment, LocalDate date) {
 | 
				
			||||||
 | 
					        Appointment appointment = getAppointmentById(idAppointment);
 | 
				
			||||||
 | 
					        appointment.setAppointmentDate(date);
 | 
				
			||||||
 | 
					        this.appointmentRepository.save(appointment);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAppointmentTime(long idAppointment, LocalTime time) {
 | 
				
			||||||
 | 
					        Appointment appointment = getAppointmentById(idAppointment);
 | 
				
			||||||
 | 
					        appointment.setAppointmentTime(time);
 | 
				
			||||||
 | 
					        this.appointmentRepository.save(appointment);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAppointmentDuration(long idAppointment, LocalTime duration) {
 | 
				
			||||||
 | 
					        Appointment appointment = getAppointmentById(idAppointment);
 | 
				
			||||||
 | 
					        appointment.setAppointmentDuration(duration);
 | 
				
			||||||
 | 
					        this.appointmentRepository.save(appointment);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAppointmentPlace(long idAppointment, String place) {
 | 
				
			||||||
 | 
					        Appointment appointment = getAppointmentById(idAppointment);
 | 
				
			||||||
 | 
					        appointment.setAppointmentPlace(place);
 | 
				
			||||||
 | 
					        this.appointmentRepository.save(appointment);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAppointmentSubject(long idAppointment, String subject) {
 | 
				
			||||||
 | 
					        Appointment appointment = getAppointmentById(idAppointment);
 | 
				
			||||||
 | 
					        appointment.setAppointmentSubject(subject);
 | 
				
			||||||
 | 
					        this.appointmentRepository.save(appointment);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateAppointmentListSectionCell(long idAppointment, SectionCell sectionCell) {
 | 
				
			||||||
 | 
					        Appointment appointment = getAppointmentById(idAppointment);
 | 
				
			||||||
 | 
					        appointment.updateListSectionCell(sectionCell);
 | 
				
			||||||
 | 
					        this.appointmentRepository.save(appointment);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Appointment updateAppointment(
 | 
					    public Appointment updateAppointment(
 | 
				
			||||||
            Long id,
 | 
					            Long id,
 | 
				
			||||||
            LocalDate appointmentDate,
 | 
					            LocalDate appointmentDate,
 | 
				
			||||||
            LocalTime appointmentTime,
 | 
					            LocalTime appointmentTime,
 | 
				
			||||||
            LocalTime appointmentDuration,
 | 
					            LocalTime appointmentDuration,
 | 
				
			||||||
            String appointmentPlace,
 | 
					            String appointmentPlace,
 | 
				
			||||||
            String appointmentSubject) {
 | 
					            String appointmentSubject,
 | 
				
			||||||
 | 
					            SectionCell sectionCell) {
 | 
				
			||||||
        Optional<Appointment> appointment = this.appointmentRepository.findById(id);
 | 
					        Optional<Appointment> appointment = this.appointmentRepository.findById(id);
 | 
				
			||||||
        if (appointment.isEmpty()) {
 | 
					        if (appointment.isEmpty()) {
 | 
				
			||||||
            logger.error("updateAppointment : No appointment found with id {}", id);
 | 
					            logger.error("updateAppointment : No appointment found with id {}", id);
 | 
				
			||||||
@@ -74,6 +112,9 @@ public class AppointmentService {
 | 
				
			|||||||
        if (appointmentSubject != null) {
 | 
					        if (appointmentSubject != null) {
 | 
				
			||||||
            appointment.get().setAppointmentSubject(appointmentSubject);
 | 
					            appointment.get().setAppointmentSubject(appointmentSubject);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (sectionCell != null) {
 | 
				
			||||||
 | 
					            appointment.get().updateListSectionCell(sectionCell);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        return this.appointmentRepository.save(appointment.get());
 | 
					        return this.appointmentRepository.save(appointment.get());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,7 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.Entrepreneur;
 | 
					import enseirb.myinpulse.model.Entrepreneur;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.MakeAppointment;
 | 
				
			||||||
import enseirb.myinpulse.model.Project;
 | 
					import enseirb.myinpulse.model.Project;
 | 
				
			||||||
import enseirb.myinpulse.repository.EntrepreneurRepository;
 | 
					import enseirb.myinpulse.repository.EntrepreneurRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -41,8 +42,52 @@ public class EntrepreneurService {
 | 
				
			|||||||
        return this.entrepreneurRepository.save(entrepreneur);
 | 
					        return this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateEntrepreneurSchool(long idEntrepreneur, String school) {
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = getEntrepreneurById(idEntrepreneur);
 | 
				
			||||||
 | 
					        entrepreneur.setSchool(school);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateEntrepreneurCourse(long idEntrepreneur, String course) {
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = getEntrepreneurById(idEntrepreneur);
 | 
				
			||||||
 | 
					        entrepreneur.setCourse(course);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateEntrepreneurSneeStatus(long idEntrepreneur, boolean status) {
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = getEntrepreneurById(idEntrepreneur);
 | 
				
			||||||
 | 
					        entrepreneur.setSneeStatus(status);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateEntrepreneurProjectParticipation(
 | 
				
			||||||
 | 
					            long idEntrepreneur, Project projectParticipation) {
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = getEntrepreneurById(idEntrepreneur);
 | 
				
			||||||
 | 
					        entrepreneur.setProjectParticipation(projectParticipation);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateEntrepreneurProjectProposed(long idEntrepreneur, Project projectProposed) {
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = getEntrepreneurById(idEntrepreneur);
 | 
				
			||||||
 | 
					        entrepreneur.setProjectParticipation(projectProposed);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateEntrepreneurMakeAppointment(
 | 
				
			||||||
 | 
					            long idEntrepreneur, MakeAppointment makeAppointment) {
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur = getEntrepreneurById(idEntrepreneur);
 | 
				
			||||||
 | 
					        entrepreneur.setMakeAppointment(makeAppointment);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(entrepreneur);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Entrepreneur updateEntrepreneur(
 | 
					    public Entrepreneur updateEntrepreneur(
 | 
				
			||||||
            Long id, String school, String course, Boolean sneeStatus) {
 | 
					            Long id,
 | 
				
			||||||
 | 
					            String school,
 | 
				
			||||||
 | 
					            String course,
 | 
				
			||||||
 | 
					            Boolean sneeStatus,
 | 
				
			||||||
 | 
					            Project projectParticipation,
 | 
				
			||||||
 | 
					            Project projectProposed,
 | 
				
			||||||
 | 
					            MakeAppointment makeAppointment) {
 | 
				
			||||||
        Optional<Entrepreneur> entrepreneur = entrepreneurRepository.findById(id);
 | 
					        Optional<Entrepreneur> entrepreneur = entrepreneurRepository.findById(id);
 | 
				
			||||||
        if (entrepreneur.isEmpty()) {
 | 
					        if (entrepreneur.isEmpty()) {
 | 
				
			||||||
            logger.error("updateEntrepreneur : No entrepreneur found with id {}", id);
 | 
					            logger.error("updateEntrepreneur : No entrepreneur found with id {}", id);
 | 
				
			||||||
@@ -58,10 +103,33 @@ public class EntrepreneurService {
 | 
				
			|||||||
        if (sneeStatus != null) {
 | 
					        if (sneeStatus != null) {
 | 
				
			||||||
            entrepreneur.get().setSneeStatus(sneeStatus);
 | 
					            entrepreneur.get().setSneeStatus(sneeStatus);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (projectParticipation != null) {
 | 
				
			||||||
 | 
					            entrepreneur.get().setProjectParticipation(projectParticipation);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (projectProposed != null) {
 | 
				
			||||||
 | 
					            entrepreneur.get().setProjectParticipation(projectProposed);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (makeAppointment != null) {
 | 
				
			||||||
 | 
					            entrepreneur.get().setMakeAppointment(makeAppointment);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        return this.entrepreneurRepository.save(entrepreneur.get());
 | 
					        return this.entrepreneurRepository.save(entrepreneur.get());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Iterable<Entrepreneur> GetEntrepreneurByProject(Project project) {
 | 
					    public Iterable<Entrepreneur> GetEntrepreneurByProject(Project project) {
 | 
				
			||||||
        return this.entrepreneurRepository.getEntrepreneurByProjectParticipation(project);
 | 
					        return this.entrepreneurRepository.getEntrepreneurByProjectParticipation(project);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void deleteEntrepreneur(Entrepreneur e) {
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.delete(e);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void validateEntrepreneurById(Long id) {
 | 
				
			||||||
 | 
					        System.out.println("\nVALIDATING\n");
 | 
				
			||||||
 | 
					        Optional<Entrepreneur> e = this.entrepreneurRepository.findById(id);
 | 
				
			||||||
 | 
					        if (e.isEmpty()) {
 | 
				
			||||||
 | 
					            throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Entrepreneur n'existe pas");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        e.get().setPending(false);
 | 
				
			||||||
 | 
					        this.entrepreneurRepository.save(e.get());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,7 +1,8 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.Administrator;
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.PENDING;
 | 
				
			||||||
import enseirb.myinpulse.model.Project;
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.*;
 | 
				
			||||||
import enseirb.myinpulse.repository.ProjectRepository;
 | 
					import enseirb.myinpulse.repository.ProjectRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.apache.logging.log4j.LogManager;
 | 
					import org.apache.logging.log4j.LogManager;
 | 
				
			||||||
@@ -11,7 +12,6 @@ import org.springframework.http.HttpStatus;
 | 
				
			|||||||
import org.springframework.stereotype.Service;
 | 
					import org.springframework.stereotype.Service;
 | 
				
			||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					import org.springframework.web.server.ResponseStatusException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import java.time.LocalDate;
 | 
					 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
import java.util.Optional;
 | 
					import java.util.Optional;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -49,12 +49,50 @@ public class ProjectService {
 | 
				
			|||||||
        return this.projectRepository.save(project);
 | 
					        return this.projectRepository.save(project);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateProjectName(long idProject, String name) {
 | 
				
			||||||
 | 
					        Project project = getProjectById(idProject);
 | 
				
			||||||
 | 
					        project.setProjectName(name);
 | 
				
			||||||
 | 
					        this.projectRepository.save(project);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateProjectLogo(long idProject, byte[] logo) {
 | 
				
			||||||
 | 
					        Project project = getProjectById(idProject);
 | 
				
			||||||
 | 
					        project.setLogo(logo);
 | 
				
			||||||
 | 
					        this.projectRepository.save(project);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateProjectStatus(long idProject, ProjectDecisionValue status) {
 | 
				
			||||||
 | 
					        Project project = getProjectById(idProject);
 | 
				
			||||||
 | 
					        project.setProjectStatus(status);
 | 
				
			||||||
 | 
					        this.projectRepository.save(project);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateProjectEntrepreneurParticipation(
 | 
				
			||||||
 | 
					            long idProject, Entrepreneur entrepreneurParticipation) {
 | 
				
			||||||
 | 
					        Project project = getProjectById(idProject);
 | 
				
			||||||
 | 
					        project.updateListEntrepreneurParticipation(entrepreneurParticipation);
 | 
				
			||||||
 | 
					        this.projectRepository.save(project);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateProjectListSectionCell(long idProject, SectionCell sectionCell) {
 | 
				
			||||||
 | 
					        Project project = getProjectById(idProject);
 | 
				
			||||||
 | 
					        project.updateListSectionCell(sectionCell);
 | 
				
			||||||
 | 
					        this.projectRepository.save(project);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateProjectAdministrator(long idProject, Administrator administrator) {
 | 
				
			||||||
 | 
					        Project project = getProjectById(idProject);
 | 
				
			||||||
 | 
					        project.setProjectAdministrator(administrator);
 | 
				
			||||||
 | 
					        this.projectRepository.save(project);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Project updateProject(
 | 
					    public Project updateProject(
 | 
				
			||||||
            Long id,
 | 
					            Long id,
 | 
				
			||||||
            String projectName,
 | 
					            String projectName,
 | 
				
			||||||
            byte[] logo,
 | 
					            byte[] logo,
 | 
				
			||||||
            LocalDate creationDate,
 | 
					            ProjectDecisionValue projectStatus,
 | 
				
			||||||
            String projectStatus,
 | 
					            Entrepreneur entrepreneurParticipation,
 | 
				
			||||||
 | 
					            SectionCell sectionCell,
 | 
				
			||||||
            Administrator administrator) {
 | 
					            Administrator administrator) {
 | 
				
			||||||
        Optional<Project> project = this.projectRepository.findById(id);
 | 
					        Optional<Project> project = this.projectRepository.findById(id);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -70,22 +108,25 @@ public class ProjectService {
 | 
				
			|||||||
        if (logo != null) {
 | 
					        if (logo != null) {
 | 
				
			||||||
            project.get().setLogo(logo);
 | 
					            project.get().setLogo(logo);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (creationDate != null) {
 | 
					 | 
				
			||||||
            project.get().setCreationDate(creationDate);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (projectStatus != null) {
 | 
					        if (projectStatus != null) {
 | 
				
			||||||
 | 
					            // TODO: check if this is really useful
 | 
				
			||||||
 | 
					            /*
 | 
				
			||||||
            if (!validateStatus(projectStatus)) {
 | 
					            if (!validateStatus(projectStatus)) {
 | 
				
			||||||
                logger.error("updateProjectStatus: Invalid status {}", projectStatus);
 | 
					                logger.error("updateProjectStatus: Invalid status {}", projectStatus);
 | 
				
			||||||
                throw new ResponseStatusException(
 | 
					                throw new ResponseStatusException(
 | 
				
			||||||
                        HttpStatus.NOT_ACCEPTABLE, "Ce status n'est pas accepté");
 | 
					                        HttpStatus.NOT_ACCEPTABLE, "Ce status n'est pas accepté");
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					            */
 | 
				
			||||||
            project.get().setProjectStatus(projectStatus);
 | 
					            project.get().setProjectStatus(projectStatus);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (entrepreneurParticipation != null) {
 | 
				
			||||||
 | 
					            project.get().updateListEntrepreneurParticipation(entrepreneurParticipation);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (sectionCell != null) {
 | 
				
			||||||
 | 
					            project.get().updateListSectionCell(sectionCell);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        if (administrator != null) {
 | 
					        if (administrator != null) {
 | 
				
			||||||
            project.get().setAdministrator(administrator);
 | 
					            project.get().setProjectAdministrator(administrator);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return this.projectRepository.save(project.get());
 | 
					        return this.projectRepository.save(project.get());
 | 
				
			||||||
@@ -96,10 +137,23 @@ public class ProjectService {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Iterable<Project> getPendingProjects() {
 | 
					    public Iterable<Project> getPendingProjects() {
 | 
				
			||||||
        return this.projectRepository.findByProjectStatus("PENDING");
 | 
					        return this.projectRepository.findByProjectStatus(PENDING);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void deleteProjectById(Long id) {
 | 
					    public void deleteProjectById(Long id) {
 | 
				
			||||||
        this.projectRepository.deleteById(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,5 +1,6 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Appointment;
 | 
				
			||||||
import enseirb.myinpulse.model.Report;
 | 
					import enseirb.myinpulse.model.Report;
 | 
				
			||||||
import enseirb.myinpulse.repository.ReportRepository;
 | 
					import enseirb.myinpulse.repository.ReportRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -46,7 +47,19 @@ public class ReportService {
 | 
				
			|||||||
        this.reportRepository.deleteById(id);
 | 
					        this.reportRepository.deleteById(id);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Report updateReport(Long id, String reportContent) {
 | 
					    public void updateReportContent(long idReport, String content) {
 | 
				
			||||||
 | 
					        Report report = getReportById(idReport);
 | 
				
			||||||
 | 
					        report.setReportContent(content);
 | 
				
			||||||
 | 
					        this.reportRepository.save(report);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateReportAppointment(long idReport, Appointment appointment) {
 | 
				
			||||||
 | 
					        Report report = getReportById(idReport);
 | 
				
			||||||
 | 
					        report.setAppointmentReport(appointment);
 | 
				
			||||||
 | 
					        this.reportRepository.save(report);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Report updateReport(Long id, String reportContent, Appointment appointment) {
 | 
				
			||||||
        Optional<Report> report = this.reportRepository.findById(id);
 | 
					        Optional<Report> report = this.reportRepository.findById(id);
 | 
				
			||||||
        if (report.isEmpty()) {
 | 
					        if (report.isEmpty()) {
 | 
				
			||||||
            logger.error("updateReport : No report found with id {}", id);
 | 
					            logger.error("updateReport : No report found with id {}", id);
 | 
				
			||||||
@@ -55,6 +68,9 @@ public class ReportService {
 | 
				
			|||||||
        if (reportContent != null) {
 | 
					        if (reportContent != null) {
 | 
				
			||||||
            report.get().setReportContent(reportContent);
 | 
					            report.get().setReportContent(reportContent);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (appointment != null) {
 | 
				
			||||||
 | 
					            report.get().setAppointmentReport(appointment);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        return this.reportRepository.save(report.get());
 | 
					        return this.reportRepository.save(report.get());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,6 @@
 | 
				
			|||||||
package enseirb.myinpulse.service.database;
 | 
					package enseirb.myinpulse.service.database;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.Annotation;
 | 
				
			||||||
import enseirb.myinpulse.model.Appointment;
 | 
					import enseirb.myinpulse.model.Appointment;
 | 
				
			||||||
import enseirb.myinpulse.model.Project;
 | 
					import enseirb.myinpulse.model.Project;
 | 
				
			||||||
import enseirb.myinpulse.model.SectionCell;
 | 
					import enseirb.myinpulse.model.SectionCell;
 | 
				
			||||||
@@ -13,7 +14,10 @@ import org.springframework.stereotype.Service;
 | 
				
			|||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					import org.springframework.web.server.ResponseStatusException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import java.time.LocalDateTime;
 | 
					import java.time.LocalDateTime;
 | 
				
			||||||
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.HashMap;
 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					import java.util.Map;
 | 
				
			||||||
import java.util.Optional;
 | 
					import java.util.Optional;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Service
 | 
					@Service
 | 
				
			||||||
@@ -50,22 +54,63 @@ public class SectionCellService {
 | 
				
			|||||||
        this.sectionCellRepository.deleteById(id);
 | 
					        this.sectionCellRepository.deleteById(id);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateSectionCellReferenceId(Long idSectionCell, Long referenceId) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = this.getSectionCellById(idSectionCell);
 | 
				
			||||||
 | 
					        sectionCell.setIdReference(referenceId);
 | 
				
			||||||
 | 
					        this.sectionCellRepository.save(sectionCell);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateSectionCellContent(long idSectionCell, String content) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = getSectionCellById(idSectionCell);
 | 
				
			||||||
 | 
					        sectionCell.setContentSectionCell(content);
 | 
				
			||||||
 | 
					        this.sectionCellRepository.save(sectionCell);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateSectionCellListAppointment(long idSectionCell, Appointment appointment) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = getSectionCellById(idSectionCell);
 | 
				
			||||||
 | 
					        sectionCell.updateAppointmentSectionCell(appointment);
 | 
				
			||||||
 | 
					        this.sectionCellRepository.save(sectionCell);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateSectionCellListAnnotation(long idSectionCell, Annotation annotation) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = getSectionCellById(idSectionCell);
 | 
				
			||||||
 | 
					        sectionCell.updateListAnnotation(annotation);
 | 
				
			||||||
 | 
					        this.sectionCellRepository.save(sectionCell);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateSectionCellProject(long idSectionCell, Project project) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = getSectionCellById(idSectionCell);
 | 
				
			||||||
 | 
					        sectionCell.setProjectSectionCell(project);
 | 
				
			||||||
 | 
					        this.sectionCellRepository.save(sectionCell);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public SectionCell updateSectionCell(
 | 
					    public SectionCell updateSectionCell(
 | 
				
			||||||
            Long id, Long sectionId, String contentSectionCell, LocalDateTime modificationDate) {
 | 
					            Long id,
 | 
				
			||||||
 | 
					            String contentSectionCell,
 | 
				
			||||||
 | 
					            Appointment appointment,
 | 
				
			||||||
 | 
					            Annotation annotation,
 | 
				
			||||||
 | 
					            Project project) {
 | 
				
			||||||
        Optional<SectionCell> sectionCell = this.sectionCellRepository.findById(id);
 | 
					        Optional<SectionCell> sectionCell = this.sectionCellRepository.findById(id);
 | 
				
			||||||
        if (sectionCell.isEmpty()) {
 | 
					        if (sectionCell.isEmpty()) {
 | 
				
			||||||
            logger.error("updateSectionCell : No sectionCell found with id {}", id);
 | 
					            logger.error("updateSectionCell : No sectionCell found with id {}", id);
 | 
				
			||||||
            throw new ResponseStatusException(
 | 
					            throw new ResponseStatusException(
 | 
				
			||||||
                    HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas");
 | 
					                    HttpStatus.NOT_FOUND, "Cette cellule de section n'existe pas");
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        if (sectionId != null) {
 | 
					 | 
				
			||||||
            sectionCell.get().setSectionId(sectionId);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
        if (contentSectionCell != null) {
 | 
					        if (contentSectionCell != null) {
 | 
				
			||||||
            sectionCell.get().setContentSectionCell(contentSectionCell);
 | 
					            sectionCell.get().setContentSectionCell(contentSectionCell);
 | 
				
			||||||
 | 
					            sectionCell.get().setModificationDate(LocalDateTime.now());
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        if (modificationDate != null) {
 | 
					        if (appointment != null) {
 | 
				
			||||||
            sectionCell.get().setModificationDate(modificationDate);
 | 
					            sectionCell.get().updateAppointmentSectionCell(appointment);
 | 
				
			||||||
 | 
					            sectionCell.get().setModificationDate(LocalDateTime.now());
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (annotation != null) {
 | 
				
			||||||
 | 
					            sectionCell.get().updateListAnnotation(annotation);
 | 
				
			||||||
 | 
					            sectionCell.get().setModificationDate(LocalDateTime.now());
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (project != null) {
 | 
				
			||||||
 | 
					            sectionCell.get().setProjectSectionCell(project);
 | 
				
			||||||
 | 
					            sectionCell.get().setModificationDate(LocalDateTime.now());
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        return this.sectionCellRepository.save(sectionCell.get());
 | 
					        return this.sectionCellRepository.save(sectionCell.get());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -74,6 +119,18 @@ public class SectionCellService {
 | 
				
			|||||||
        return this.sectionCellRepository.findByProjectSectionCellAndSectionId(project, sectionId);
 | 
					        return this.sectionCellRepository.findByProjectSectionCellAndSectionId(project, sectionId);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<SectionCell> getSectionCellsByProject(Project project) {
 | 
				
			||||||
 | 
					        logger.info("Fetching SectionCells for Project ID: {}", project.getIdProject());
 | 
				
			||||||
 | 
					        Iterable<SectionCell> sectionCells =
 | 
				
			||||||
 | 
					                this.sectionCellRepository.findByProjectSectionCell(project);
 | 
				
			||||||
 | 
					        List<SectionCell> sectionCellList = new ArrayList<>();
 | 
				
			||||||
 | 
					        sectionCells.forEach(
 | 
				
			||||||
 | 
					                cell -> {
 | 
				
			||||||
 | 
					                    sectionCellList.add(cell);
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					        return sectionCellList;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public Long getProjectId(Long sectionCellId) {
 | 
					    public Long getProjectId(Long sectionCellId) {
 | 
				
			||||||
        SectionCell sectionCell = getSectionCellById(sectionCellId);
 | 
					        SectionCell sectionCell = getSectionCellById(sectionCellId);
 | 
				
			||||||
        Project sectionProject = sectionCell.getProjectSectionCell();
 | 
					        Project sectionProject = sectionCell.getProjectSectionCell();
 | 
				
			||||||
@@ -90,4 +147,37 @@ public class SectionCellService {
 | 
				
			|||||||
        return sectionCellRepository.findByProjectSectionCellAndSectionIdAndModificationDateBefore(
 | 
					        return sectionCellRepository.findByProjectSectionCellAndSectionIdAndModificationDateBefore(
 | 
				
			||||||
                project, sectionId, date);
 | 
					                project, sectionId, date);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<SectionCell> getLatestSectionCellsByIdReferenceBeforeDate(
 | 
				
			||||||
 | 
					            Project project, long sectionId, LocalDateTime date) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 1. Fetch ALL relevant SectionCells modified before the date
 | 
				
			||||||
 | 
					        Iterable<SectionCell> allMatchingCells =
 | 
				
			||||||
 | 
					                sectionCellRepository.findByProjectSectionCellAndSectionIdAndModificationDateBefore(
 | 
				
			||||||
 | 
					                        project, sectionId, date);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 2. Find the latest for each idReference
 | 
				
			||||||
 | 
					        Map<Long, SectionCell> latestCellsByIdReference = new HashMap<>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (SectionCell cell : allMatchingCells) {
 | 
				
			||||||
 | 
					            Long idReference = cell.getIdReference();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            // Check if we've seen this idReference before
 | 
				
			||||||
 | 
					            if (latestCellsByIdReference.containsKey(idReference)) {
 | 
				
			||||||
 | 
					                // If yes, compare modification dates
 | 
				
			||||||
 | 
					                SectionCell existingLatest = latestCellsByIdReference.get(idReference);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                // If the current cell is more recent, update the map
 | 
				
			||||||
 | 
					                if (cell.getModificationDate().isAfter(existingLatest.getModificationDate())) {
 | 
				
			||||||
 | 
					                    latestCellsByIdReference.put(idReference, cell);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                // If this is the first time we see this idReference, add it to the map
 | 
				
			||||||
 | 
					                latestCellsByIdReference.put(idReference, cell);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 3. Return the collection of the latest cells (the values from the map)
 | 
				
			||||||
 | 
					        return latestCellsByIdReference.values();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -30,6 +30,15 @@ public class UserService {
 | 
				
			|||||||
        return this.userRepository.findAll();
 | 
					        return this.userRepository.findAll();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public User getUserById(long id) {
 | 
				
			||||||
 | 
					        Optional<User> user = this.userRepository.findById(id);
 | 
				
			||||||
 | 
					        if (user.isEmpty()) {
 | 
				
			||||||
 | 
					            logger.error("getUserById : No user found with id {}", id);
 | 
				
			||||||
 | 
					            throw new ResponseStatusException(HttpStatus.NOT_FOUND, "Cet utilisateur n'existe pas");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return user.get();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // TODO
 | 
					    // TODO
 | 
				
			||||||
    public User getUserByEmail(String email) {
 | 
					    public User getUserByEmail(String email) {
 | 
				
			||||||
        Optional<User> opt_user = this.userRepository.findByPrimaryMail(email);
 | 
					        Optional<User> opt_user = this.userRepository.findByPrimaryMail(email);
 | 
				
			||||||
@@ -49,11 +58,41 @@ public class UserService {
 | 
				
			|||||||
        return this.userRepository.save(user);
 | 
					        return this.userRepository.save(user);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateUserSurname(long idUser, String surname) {
 | 
				
			||||||
 | 
					        User user = getUserById(idUser);
 | 
				
			||||||
 | 
					        user.setUserSurname(surname);
 | 
				
			||||||
 | 
					        this.userRepository.save(user);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateUserName(long idUser, String name) {
 | 
				
			||||||
 | 
					        User user = getUserById(idUser);
 | 
				
			||||||
 | 
					        user.setUserName(name);
 | 
				
			||||||
 | 
					        this.userRepository.save(user);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateUserPrimaryMail(long idUser, String primaryMail) {
 | 
				
			||||||
 | 
					        User user = getUserById(idUser);
 | 
				
			||||||
 | 
					        user.setPrimaryMail(primaryMail);
 | 
				
			||||||
 | 
					        this.userRepository.save(user);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateUserSecondaryMail(long idUser, String secondaryMail) {
 | 
				
			||||||
 | 
					        User user = getUserById(idUser);
 | 
				
			||||||
 | 
					        user.setSecondaryMail(secondaryMail);
 | 
				
			||||||
 | 
					        this.userRepository.save(user);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void updateUserPhoneNumber(long idUser, String phoneNumber) {
 | 
				
			||||||
 | 
					        User user = getUserById(idUser);
 | 
				
			||||||
 | 
					        user.setPhoneNumber(phoneNumber);
 | 
				
			||||||
 | 
					        this.userRepository.save(user);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public User updateUser(
 | 
					    public User updateUser(
 | 
				
			||||||
            @PathVariable Long id,
 | 
					            @PathVariable Long id,
 | 
				
			||||||
            String userSurname,
 | 
					            String userSurname,
 | 
				
			||||||
            String userName,
 | 
					            String userName,
 | 
				
			||||||
            String mainMail,
 | 
					            String primaryMail,
 | 
				
			||||||
            String secondaryMail,
 | 
					            String secondaryMail,
 | 
				
			||||||
            String phoneNumber) {
 | 
					            String phoneNumber) {
 | 
				
			||||||
        Optional<User> user = userRepository.findById(id);
 | 
					        Optional<User> user = userRepository.findById(id);
 | 
				
			||||||
@@ -67,8 +106,8 @@ public class UserService {
 | 
				
			|||||||
        if (userSurname != null) {
 | 
					        if (userSurname != null) {
 | 
				
			||||||
            user.get().setUserSurname(userSurname);
 | 
					            user.get().setUserSurname(userSurname);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        if (mainMail != null) {
 | 
					        if (primaryMail != null) {
 | 
				
			||||||
            user.get().setPrimaryMail(mainMail);
 | 
					            user.get().setPrimaryMail(primaryMail);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        if (secondaryMail != null) {
 | 
					        if (secondaryMail != null) {
 | 
				
			||||||
            user.get().setSecondaryMail(secondaryMail);
 | 
					            user.get().setSecondaryMail(secondaryMail);
 | 
				
			||||||
@@ -78,4 +117,8 @@ public class UserService {
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
        return this.userRepository.save(user.get());
 | 
					        return this.userRepository.save(user.get());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public Iterable<User> getPendingAccounts() {
 | 
				
			||||||
 | 
					        return this.userRepository.findAllByPendingEquals(true);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,6 @@
 | 
				
			|||||||
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/${VITE_KEYCLOAK_REALM}/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/${VITE_KEYCLOAK_REALM}
 | 
				
			||||||
spring.datasource.url=jdbc:postgresql://${DATABASE_URL}/${BACKEND_DB}
 | 
					spring.datasource.url=jdbc:postgresql://${DATABASE_URL}/${BACKEND_DB}
 | 
				
			||||||
spring.datasource.username=${BACKEND_USER}
 | 
					spring.datasource.username=${BACKEND_USER}
 | 
				
			||||||
spring.datasource.password=${BACKEND_PASSWORD}
 | 
					spring.datasource.password=${BACKEND_PASSWORD}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1,13 @@
 | 
				
			|||||||
 | 
					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.issuer-uri=http://localhost:7080/realms/test
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					logging.pattern.console=%d{yyyy-MMM-dd HH:mm:ss.SSS} [%thread] %highlight(%-5level) %cyan(%logger{15}) - %msg %n
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					spring.datasource.url=jdbc:h2:mem:testdb;DB_CLOSE_DELAY=-1
 | 
				
			||||||
 | 
					spring.datasource.driverClassName=org.h2.Driver
 | 
				
			||||||
 | 
					spring.datasource.username=sa
 | 
				
			||||||
 | 
					spring.datasource.password=
 | 
				
			||||||
 | 
					spring.jpa.database-platform=org.hibernate.dialect.H2Dialect
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					spring.jpa.hibernate.ddl-auto=create
 | 
				
			||||||
@@ -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;
 | 
					 | 
				
			||||||
							
								
								
									
										92
									
								
								MyINPulse-back/src/main/resources/import.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								MyINPulse-back/src/main/resources/import.sql
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,92 @@
 | 
				
			|||||||
 | 
					-- Initial Database State Script
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Administrators
 | 
				
			||||||
 | 
					INSERT INTO administrator (idAdministrator) VALUES
 | 
				
			||||||
 | 
					(1),
 | 
				
			||||||
 | 
					(2),
 | 
				
			||||||
 | 
					(3); -- Added more administrators
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert User Inpulse (some pending)
 | 
				
			||||||
 | 
					INSERT INTO user_inpulse (idUser, userSurname, userName, primaryMail, secondaryMail, phoneNumber, pending) VALUES
 | 
				
			||||||
 | 
					(1, 'Doe', 'John', 'john.doe@example.com', NULL, '123-456-7890', FALSE),
 | 
				
			||||||
 | 
					(2, 'Smith', 'Jane', 'jane.smith@example.com', 'jane.s@altmail.com', '987-654-3210', FALSE),
 | 
				
			||||||
 | 
					(3, 'Williams', 'Peter', 'peter.w@example.com', NULL, NULL, TRUE), -- Pending user
 | 
				
			||||||
 | 
					(4, 'Jones', 'Mary', 'mary.j@example.com', NULL, '555-123-4567', FALSE),
 | 
				
			||||||
 | 
					(5, 'Brown', 'Michael', 'michael.b@example.com', 'mike.brown@work.com', '111-222-3333', FALSE),
 | 
				
			||||||
 | 
					(6, 'Garcia', 'Maria', 'maria.g@example.com', NULL, '444-555-6666', TRUE), -- Another pending user
 | 
				
			||||||
 | 
					(7, 'Miller', 'David', 'david.m@example.com', NULL, '777-888-9999', FALSE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Entrepreneurs
 | 
				
			||||||
 | 
					INSERT INTO entrepreneur (idEntrepreneur, school, course, sneeStatus, idProjectParticipation, idProjectProposed, idMakeAppointment) VALUES
 | 
				
			||||||
 | 
					(1, 'Business School A', 'MBA', TRUE, NULL, NULL, NULL),
 | 
				
			||||||
 | 
					(2, 'Tech University B', 'Computer Science', FALSE, NULL, NULL, NULL),
 | 
				
			||||||
 | 
					(3, 'Art Institute C', 'Graphic Design', TRUE, NULL, NULL, NULL),
 | 
				
			||||||
 | 
					(4, 'Science College D', 'Biology', FALSE, NULL, NULL, NULL),
 | 
				
			||||||
 | 
					(5, 'Engineering School E', 'Mechanical Engineering', TRUE, NULL, NULL, NULL); -- Added more entrepreneurs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Projects
 | 
				
			||||||
 | 
					-- Main project
 | 
				
			||||||
 | 
					INSERT INTO project (IdProject, projectName, loga, creationDate, projectStatus, pending, idAdministrator, entrepreneurProposed) VALUES
 | 
				
			||||||
 | 
					(101, 'Innovative Startup Idea', NULL, '2023-10-26', 'In Progress', FALSE, 1, NULL),
 | 
				
			||||||
 | 
					(102, 'Pending Project Alpha', NULL, '2024-01-15', 'Planning', TRUE, 1, NULL), -- Pending project
 | 
				
			||||||
 | 
					(103, 'Pending Project Beta', NULL, '2024-02-20', 'Idea Stage', TRUE, NULL, 1), -- Another pending project, proposed by entrepreneur 1
 | 
				
			||||||
 | 
					(104, 'E-commerce Platform Development', NULL, '2024-03-10', 'Completed', FALSE, 2, NULL), -- Completed project
 | 
				
			||||||
 | 
					(105, 'Mobile App for Education', NULL, '2024-04-01', 'In Progress', FALSE, 3, NULL),
 | 
				
			||||||
 | 
					(106, 'Pending Research Proposal', NULL, '2024-04-25', 'Drafting', TRUE, NULL, 4); -- Pending project proposed by entrepreneur 4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Link Entrepreneurs to projects (Project Participation and Proposed)
 | 
				
			||||||
 | 
					-- Based on the current schema, we'll update the entrepreneur table directly.
 | 
				
			||||||
 | 
					-- This might need adjustment based on actual application logic if it's a many-to-many.
 | 
				
			||||||
 | 
					UPDATE entrepreneur SET idProjectParticipation = 101 WHERE idEntrepreneur IN (1, 2); -- Entrepreneurs 1 and 2 participate in Project 101
 | 
				
			||||||
 | 
					UPDATE entrepreneur SET idProjectParticipation = 104 WHERE idEntrepreneur = 3; -- Entrepreneur 3 participates in Project 104
 | 
				
			||||||
 | 
					UPDATE entrepreneur SET idProjectProposed = 103 WHERE idEntrepreneur = 1; -- Entrepreneur 1 proposed Project 103
 | 
				
			||||||
 | 
					UPDATE entrepreneur SET idProjectProposed = 106 WHERE idEntrepreneur = 4; -- Entrepreneur 4 proposed Project 106
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Section Cells for the main project (Project 101) and other projects
 | 
				
			||||||
 | 
					INSERT INTO section_cell (idSectionCell, IdReference, sectionid, contentSectionCell, modificationDate, idProject) VALUES
 | 
				
			||||||
 | 
					(1001, NULL, 1, 'Initial project description for Project 101.', '2023-10-26 10:00:00', 101),
 | 
				
			||||||
 | 
					(1002, 1001, 2, 'Market analysis summary for Project 101.', '2023-10-27 14:30:00', 101),
 | 
				
			||||||
 | 
					(1003, NULL, 3, 'Team member profiles for Project 101.', '2023-10-28 09:00:00', 101),
 | 
				
			||||||
 | 
					(1004, NULL, 1, 'Project brief for Project 104.', '2024-03-10 11:00:00', 104),
 | 
				
			||||||
 | 
					(1005, 1004, 2, 'Technical specifications for Project 104.', '2024-03-15 16:00:00', 104),
 | 
				
			||||||
 | 
					(1006, NULL, 1, 'Initial concept for Project 105.', '2024-04-01 09:30:00', 105);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Appointments
 | 
				
			||||||
 | 
					INSERT INTO appointment (idAppointment, appointmentDate, appointmentTime, appointmentDuration, appointmentPlace, appointmentSubject) VALUES
 | 
				
			||||||
 | 
					(2001, '2023-11-05', '11:00:00', '01:00:00', 'Meeting Room A', 'Project 101 Kick-off Meeting'),
 | 
				
			||||||
 | 
					(2002, '2023-11-10', '14:00:00', '00:30:00', 'Online', 'Project 101 Follow-up Discussion'),
 | 
				
			||||||
 | 
					(2003, '2024-03-20', '10:00:00', '01:30:00', 'Client Office', 'Project 104 Final Review'),
 | 
				
			||||||
 | 
					(2004, '2024-04-10', '15:00:00', '00:45:00', 'Video Call', 'Project 105 Initial Sync'); -- Added more appointments
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Concerns (linking Appointments and Section Cells)
 | 
				
			||||||
 | 
					INSERT INTO concern (IdAppointment, idSectionCell) VALUES
 | 
				
			||||||
 | 
					(2001, 1001), -- Kick-off meeting concerns section 1001 (Project 101)
 | 
				
			||||||
 | 
					(2001, 1002), -- Kick-off meeting concerns section 1002 (Project 101)
 | 
				
			||||||
 | 
					(2002, 1002), -- Follow-up concerns section 1002 (Project 101)
 | 
				
			||||||
 | 
					(2003, 1004), -- Project 104 review concerns section 1004
 | 
				
			||||||
 | 
					(2003, 1005), -- Project 104 review concerns section 1005
 | 
				
			||||||
 | 
					(2004, 1006); -- Project 105 sync concerns section 1006
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Make Appointments (linking Appointments, Administrators, and Entrepreneurs)
 | 
				
			||||||
 | 
					INSERT INTO make_appointment (idMakeAppointment, idAppointment, idAdministrator, idEntrepreneur) VALUES
 | 
				
			||||||
 | 
					(3001, 2001, 1, 1), -- Admin 1 scheduled appointment 2001 with Entrepreneur 1
 | 
				
			||||||
 | 
					(3002, 2001, 1, 2), -- Admin 1 scheduled appointment 2001 with Entrepreneur 2
 | 
				
			||||||
 | 
					(3003, 2002, 1, 1), -- Admin 1 scheduled appointment 2002 with Entrepreneur 1
 | 
				
			||||||
 | 
					(3004, 2003, 2, 3), -- Admin 2 scheduled appointment 2003 with Entrepreneur 3
 | 
				
			||||||
 | 
					(3005, 2004, 3, 5); -- Admin 3 scheduled appointment 2004 with Entrepreneur 5
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Annotations (linking to Section Cells and Administrators)
 | 
				
			||||||
 | 
					INSERT INTO annotation (IdAnnotation, comment, idSectionCell, idAdministrator) VALUES
 | 
				
			||||||
 | 
					(4001, 'Needs more detail on market size.', 1002, 1),
 | 
				
			||||||
 | 
					(4002, 'Looks good.', 1001, 1),
 | 
				
			||||||
 | 
					(4003, 'Confirm technical requirements.', 1005, 2),
 | 
				
			||||||
 | 
					(4004, 'Initial thoughts on UI/UX.', 1006, 3); -- Added more annotations
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- Insert Reports (linking to Make Appointments)
 | 
				
			||||||
 | 
					INSERT INTO report (IdReport, reportContent, idMakeAppointment) VALUES
 | 
				
			||||||
 | 
					(5001, 'Discussed project scope and timelines for Project 101.', 3001),
 | 
				
			||||||
 | 
					(5002, 'Reviewed market analysis feedback for Project 101.', 3003),
 | 
				
			||||||
 | 
					(5003, 'Final sign-off on Project 104 deliverables.', 3004),
 | 
				
			||||||
 | 
					(5004, 'Discussed initial concepts for Project 105.', 3005); -- Added more reports
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					-- The project ID for the main project is 101.
 | 
				
			||||||
@@ -1,12 +1,13 @@
 | 
				
			|||||||
package enseirb.myinpulse;
 | 
					package enseirb.myinpulse;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.*;
 | 
				
			||||||
import static org.junit.jupiter.api.Assertions.assertThrows;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
import enseirb.myinpulse.model.Administrator;
 | 
					import static org.junit.jupiter.api.Assertions.*;
 | 
				
			||||||
import enseirb.myinpulse.model.Project;
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.*;
 | 
				
			||||||
import enseirb.myinpulse.service.AdminApiService;
 | 
					import enseirb.myinpulse.service.AdminApiService;
 | 
				
			||||||
import enseirb.myinpulse.service.database.AdministratorService;
 | 
					import enseirb.myinpulse.service.UtilsService;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.database.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.junit.jupiter.api.BeforeAll;
 | 
					import org.junit.jupiter.api.BeforeAll;
 | 
				
			||||||
import org.junit.jupiter.api.Test;
 | 
					import org.junit.jupiter.api.Test;
 | 
				
			||||||
@@ -15,31 +16,105 @@ import org.springframework.boot.test.context.SpringBootTest;
 | 
				
			|||||||
import org.springframework.transaction.annotation.Transactional;
 | 
					import org.springframework.transaction.annotation.Transactional;
 | 
				
			||||||
import org.springframework.web.server.ResponseStatusException;
 | 
					import org.springframework.web.server.ResponseStatusException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.time.LocalDate;
 | 
				
			||||||
 | 
					import java.time.LocalDateTime;
 | 
				
			||||||
 | 
					import java.time.LocalTime;
 | 
				
			||||||
import java.util.ArrayList;
 | 
					import java.util.ArrayList;
 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@SpringBootTest
 | 
					@SpringBootTest
 | 
				
			||||||
@Transactional
 | 
					@Transactional
 | 
				
			||||||
public class AdminApiServiceTest {
 | 
					public class AdminApiServiceTest {
 | 
				
			||||||
 | 
					    private static long administratorid;
 | 
				
			||||||
 | 
					    private static Administrator administrator;
 | 
				
			||||||
 | 
					    private static Entrepreneur entrepreneur;
 | 
				
			||||||
 | 
					    private static Appointment appt;
 | 
				
			||||||
 | 
					    private static Project p;
 | 
				
			||||||
    @Autowired private AdminApiService adminApiService;
 | 
					    @Autowired private AdminApiService adminApiService;
 | 
				
			||||||
 | 
					    @Autowired private ProjectService projectService;
 | 
				
			||||||
 | 
					    @Autowired private EntrepreneurService entrepreneurService;
 | 
				
			||||||
 | 
					    @Autowired private SectionCellService sectionCellService;
 | 
				
			||||||
 | 
					    @Autowired private AppointmentService appointmentService;
 | 
				
			||||||
 | 
					    @Autowired private UtilsService utilsService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @BeforeAll
 | 
					    @BeforeAll
 | 
				
			||||||
    static void setup(@Autowired AdministratorService administratorService) {
 | 
					    static void setup(
 | 
				
			||||||
 | 
					            @Autowired AdministratorService administratorService,
 | 
				
			||||||
 | 
					            @Autowired ProjectService projectService,
 | 
				
			||||||
 | 
					            @Autowired EntrepreneurService entrepreneurService,
 | 
				
			||||||
 | 
					            @Autowired AppointmentService appoitmentService,
 | 
				
			||||||
 | 
					            @Autowired SectionCellService sectionCellService) {
 | 
				
			||||||
        administratorService.addAdministrator(
 | 
					        administratorService.addAdministrator(
 | 
				
			||||||
                new Administrator(
 | 
					                new Administrator(
 | 
				
			||||||
                        "admin", "admin", "testAdmin@example.com", "testAdmin@example.com", ""));
 | 
					                        "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);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Entrepreneur entrepreneur2 =
 | 
				
			||||||
 | 
					                new Entrepreneur(
 | 
				
			||||||
 | 
					                        "GDProjets", "", "Entrepreneur2@inpulse.com", "", "", "", "info ofc", true);
 | 
				
			||||||
 | 
					        entrepreneurService.addEntrepreneur(entrepreneur2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        p =
 | 
				
			||||||
 | 
					                projectService.addNewProject(
 | 
				
			||||||
 | 
					                        new Project(
 | 
				
			||||||
 | 
					                                "sampleProjectAdminApiService",
 | 
				
			||||||
 | 
					                                null,
 | 
				
			||||||
 | 
					                                LocalDate.now(),
 | 
				
			||||||
 | 
					                                ACTIVE,
 | 
				
			||||||
 | 
					                                administratorService.getAdministratorByPrimaryMain(
 | 
				
			||||||
 | 
					                                        "testAdminFull@example.com")));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        entrepreneurService.updateEntrepreneurProjectParticipation(entrepreneur2.getIdUser(), p);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        appt =
 | 
				
			||||||
 | 
					                new Appointment(
 | 
				
			||||||
 | 
					                        null,
 | 
				
			||||||
 | 
					                        LocalDate.now(),
 | 
				
			||||||
 | 
					                        LocalTime.now(),
 | 
				
			||||||
 | 
					                        LocalTime.now(),
 | 
				
			||||||
 | 
					                        "Salle TD 03",
 | 
				
			||||||
 | 
					                        "Discussion importante");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private <T> List<T> IterableToList(Iterable<T> iterable) {
 | 
				
			||||||
 | 
					        List<T> l = new ArrayList<>();
 | 
				
			||||||
 | 
					        iterable.forEach(l::add);
 | 
				
			||||||
 | 
					        return l;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Test
 | 
					    @Test
 | 
				
			||||||
    void getProjectOfAdminIsEmpty() throws Exception {
 | 
					    void getProjectOfAdminIsEmpty() {
 | 
				
			||||||
        Iterable<Project> projects = adminApiService.getProjectsOfAdmin("testAdmin@example.com");
 | 
					        Iterable<Project> projects =
 | 
				
			||||||
        List<Project> l = new ArrayList<>();
 | 
					                adminApiService.getProjectsOfAdmin("testAdminEmpty@example.com");
 | 
				
			||||||
        projects.forEach(l::add);
 | 
					        assertEquals(0, IterableToList(projects).size());
 | 
				
			||||||
        assertEquals(0, l.size());
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Test
 | 
					    @Test
 | 
				
			||||||
    void getProjectOfInexistantAdminFails() throws Exception {
 | 
					    void getProjectOfInexistantAdminFails() {
 | 
				
			||||||
        String nonExistentAdminEmail = "testInexistantAdmin@example.com";
 | 
					        String nonExistentAdminEmail = "testInexistantAdmin@example.com";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        assertThrows(
 | 
					        assertThrows(
 | 
				
			||||||
@@ -48,4 +123,209 @@ public class AdminApiServiceTest {
 | 
				
			|||||||
                    adminApiService.getProjectsOfAdmin(nonExistentAdminEmail);
 | 
					                    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("sampleProjectAdminApiService", p.getProjectName());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @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("addProjectToAdmin", null, LocalDate.now(), ACTIVE, administrator);
 | 
				
			||||||
 | 
					        this.adminApiService.addNewProject(p1);
 | 
				
			||||||
 | 
					        assertEquals(1, administrator.getListProject().size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void addProjectToUser() {
 | 
				
			||||||
 | 
					        assertNull(entrepreneur.getProjectParticipation());
 | 
				
			||||||
 | 
					        Project p1 =
 | 
				
			||||||
 | 
					                new Project("addProjectToAdmin", 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("addProjectToAdmin", 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));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // We could do a delete active project, but it's not really useful.
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void deletePendingProject() {
 | 
				
			||||||
 | 
					        int oldsize = IterableToList(this.adminApiService.getPendingProjects()).size();
 | 
				
			||||||
 | 
					        Project p1 =
 | 
				
			||||||
 | 
					                new Project("PendingProjectAdminApiService2", null, LocalDate.now(), PENDING, null);
 | 
				
			||||||
 | 
					        Project p2 = this.adminApiService.addNewProject(p1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(oldsize + 1, IterableToList(this.adminApiService.getPendingProjects()).size());
 | 
				
			||||||
 | 
					        this.adminApiService.deleteProject(p2.getIdProject());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(oldsize, IterableToList(this.adminApiService.getPendingProjects()).size());
 | 
				
			||||||
 | 
					        for (int i = 0; i < oldsize; i++) {
 | 
				
			||||||
 | 
					            assertNotEquals(
 | 
				
			||||||
 | 
					                    p1.getIdProject(),
 | 
				
			||||||
 | 
					                    IterableToList(this.adminApiService.getPendingProjects())
 | 
				
			||||||
 | 
					                            .get(i)
 | 
				
			||||||
 | 
					                            .getIdProject());
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void getUpcommingAppointmentUnkwnownUser() {
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> {
 | 
				
			||||||
 | 
					                    Iterable<Appointment> a =
 | 
				
			||||||
 | 
					                            this.adminApiService.getUpcomingAppointments(
 | 
				
			||||||
 | 
					                                    "entrepreneur-inexistent@mail.fr");
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void getUpcommingAppointmentNoProject() {
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> {
 | 
				
			||||||
 | 
					                    Iterable<Appointment> a =
 | 
				
			||||||
 | 
					                            this.adminApiService.getUpcomingAppointments(
 | 
				
			||||||
 | 
					                                    "Entrepreneur@inpulse.com");
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void getUpcommingAppointmentEmpty() {
 | 
				
			||||||
 | 
					        Iterable<Appointment> a =
 | 
				
			||||||
 | 
					                this.adminApiService.getUpcomingAppointments("Entrepreneur2@inpulse.com");
 | 
				
			||||||
 | 
					        assertEquals(0, IterableToList(a).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void validateEntrepreneurAccount() {
 | 
				
			||||||
 | 
					        assertTrue(entrepreneurService.getEntrepreneurById(entrepreneur.getIdUser()).isPending());
 | 
				
			||||||
 | 
					        assertEquals(2, IterableToList(adminApiService.getPendingUsers()).size());
 | 
				
			||||||
 | 
					        adminApiService.validateEntrepreneurAccount(entrepreneur.getIdUser(), "");
 | 
				
			||||||
 | 
					        assertFalse(entrepreneurService.getEntrepreneurById(entrepreneur.getIdUser()).isPending());
 | 
				
			||||||
 | 
					        assertEquals(1, IterableToList(adminApiService.getPendingUsers()).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testCreateApptRepport() {
 | 
				
			||||||
 | 
					        System.err.println(appt.getIdAppointment());
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        new SectionCell(null, 1L, "jaja", LocalDateTime.now(), p));
 | 
				
			||||||
 | 
					        appointmentService.addNewAppointment(appt);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        appointmentService.updateAppointmentListSectionCell(appt.getIdAppointment(), s);
 | 
				
			||||||
 | 
					        projectService.updateProjectListSectionCell(p.getIdProject(), s);
 | 
				
			||||||
 | 
					        this.adminApiService.createAppointmentReport(
 | 
				
			||||||
 | 
					                appt.getIdAppointment(),
 | 
				
			||||||
 | 
					                new Report(null, "je rapporte de fou"),
 | 
				
			||||||
 | 
					                "testAdminFull@example.com");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testSetAdmin() {
 | 
				
			||||||
 | 
					        assertFalse(utilsService.isAnAdmin(entrepreneur.getPrimaryMail()));
 | 
				
			||||||
 | 
					        adminApiService.setAdmin(entrepreneur.getIdUser(), "");
 | 
				
			||||||
 | 
					        assertTrue(utilsService.isAnAdmin(entrepreneur.getPrimaryMail()));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1,324 @@
 | 
				
			|||||||
 | 
					package enseirb.myinpulse;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import static org.junit.jupiter.api.Assertions.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.*;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.EntrepreneurApiService;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.database.*;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					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.time.LocalDateTime;
 | 
				
			||||||
 | 
					import java.time.LocalTime;
 | 
				
			||||||
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@SpringBootTest
 | 
				
			||||||
 | 
					@Transactional
 | 
				
			||||||
 | 
					public class EntrepreneurApiServiceTest {
 | 
				
			||||||
 | 
					    private static Entrepreneur entrepreneur;
 | 
				
			||||||
 | 
					    private static Project project;
 | 
				
			||||||
 | 
					    private static Iterable<SectionCell> sectionCells2;
 | 
				
			||||||
 | 
					    private static Iterable<SectionCell> sectionCells3;
 | 
				
			||||||
 | 
					    @Autowired private EntrepreneurApiService entrepreneurApiService;
 | 
				
			||||||
 | 
					    @Autowired private EntrepreneurService entrepreneurService;
 | 
				
			||||||
 | 
					    @Autowired private ProjectService projectService;
 | 
				
			||||||
 | 
					    @Autowired private SectionCellService sectionCellService;
 | 
				
			||||||
 | 
					    @Autowired private AnnotationService annotationService;
 | 
				
			||||||
 | 
					    @Autowired private AppointmentService appointmentService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @BeforeAll
 | 
				
			||||||
 | 
					    static void setup(
 | 
				
			||||||
 | 
					            @Autowired EntrepreneurService entrepreneurService,
 | 
				
			||||||
 | 
					            @Autowired ProjectService projectService,
 | 
				
			||||||
 | 
					            @Autowired SectionCellService sectionCellService) {
 | 
				
			||||||
 | 
					        entrepreneur =
 | 
				
			||||||
 | 
					                entrepreneurService.addEntrepreneur(
 | 
				
			||||||
 | 
					                        new Entrepreneur(
 | 
				
			||||||
 | 
					                                "entre",
 | 
				
			||||||
 | 
					                                "preneur",
 | 
				
			||||||
 | 
					                                "entrepreneur@mail.fr",
 | 
				
			||||||
 | 
					                                "entrepreneur2@mail.fr",
 | 
				
			||||||
 | 
					                                "01 45 71 25 48",
 | 
				
			||||||
 | 
					                                "ENSEIRB",
 | 
				
			||||||
 | 
					                                "Info",
 | 
				
			||||||
 | 
					                                false));
 | 
				
			||||||
 | 
					        entrepreneurService.addEntrepreneur(
 | 
				
			||||||
 | 
					                new Entrepreneur(
 | 
				
			||||||
 | 
					                        "entre2",
 | 
				
			||||||
 | 
					                        "preneur2",
 | 
				
			||||||
 | 
					                        "testentrepreneur@mail.fr",
 | 
				
			||||||
 | 
					                        "testentrepreneur2@mail.fr",
 | 
				
			||||||
 | 
					                        "",
 | 
				
			||||||
 | 
					                        "ENSEGID",
 | 
				
			||||||
 | 
					                        "",
 | 
				
			||||||
 | 
					                        true));
 | 
				
			||||||
 | 
					        project =
 | 
				
			||||||
 | 
					                projectService.addNewProject(
 | 
				
			||||||
 | 
					                        new Project("Project", null, LocalDate.now(), ACTIVE, null, entrepreneur));
 | 
				
			||||||
 | 
					        entrepreneurService.updateEntrepreneurProjectProposed(entrepreneur.getIdUser(), project);
 | 
				
			||||||
 | 
					        entrepreneurService.updateEntrepreneurProjectParticipation(
 | 
				
			||||||
 | 
					                entrepreneur.getIdUser(), project);
 | 
				
			||||||
 | 
					        SectionCell s1 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        new SectionCell(
 | 
				
			||||||
 | 
					                                null,
 | 
				
			||||||
 | 
					                                2L,
 | 
				
			||||||
 | 
					                                "contenu très intéressant",
 | 
				
			||||||
 | 
					                                LocalDateTime.now(),
 | 
				
			||||||
 | 
					                                project));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        SectionCell s2 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        new SectionCell(
 | 
				
			||||||
 | 
					                                null,
 | 
				
			||||||
 | 
					                                3L,
 | 
				
			||||||
 | 
					                                "contenu très intéressant2",
 | 
				
			||||||
 | 
					                                LocalDateTime.now(),
 | 
				
			||||||
 | 
					                                project));
 | 
				
			||||||
 | 
					        sectionCells2 = sectionCellService.getSectionCellsByProject(project, 2L);
 | 
				
			||||||
 | 
					        sectionCells3 = sectionCellService.getSectionCellsByProject(project, 3L);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private <T> List<T> IterableToList(Iterable<T> iterable) {
 | 
				
			||||||
 | 
					        List<T> l = new ArrayList<>();
 | 
				
			||||||
 | 
					        iterable.forEach(l::add);
 | 
				
			||||||
 | 
					        return l;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void editValidSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("editValidSectionCell : ");
 | 
				
			||||||
 | 
					        SectionCell modified = IterableToList(sectionCells2).getLast();
 | 
				
			||||||
 | 
					        this.sectionCellService.updateSectionCellListAnnotation(
 | 
				
			||||||
 | 
					                modified.getIdSectionCell(),
 | 
				
			||||||
 | 
					                annotationService.addNewAnnotation(new Annotation(null, "oui j'annote encore")));
 | 
				
			||||||
 | 
					        this.sectionCellService.updateSectionCellListAppointment(
 | 
				
			||||||
 | 
					                modified.getIdSectionCell(),
 | 
				
			||||||
 | 
					                appointmentService.addNewAppointment(
 | 
				
			||||||
 | 
					                        new Appointment(
 | 
				
			||||||
 | 
					                                null,
 | 
				
			||||||
 | 
					                                LocalDate.now(),
 | 
				
			||||||
 | 
					                                LocalTime.now(),
 | 
				
			||||||
 | 
					                                LocalTime.of(2, 5),
 | 
				
			||||||
 | 
					                                "TD14",
 | 
				
			||||||
 | 
					                                "clément s'est plaint")));
 | 
				
			||||||
 | 
					        entrepreneurApiService.editSectionCell(
 | 
				
			||||||
 | 
					                modified.getIdSectionCell(), "modified content", "entrepreneur@mail.fr");
 | 
				
			||||||
 | 
					        // We get the data from the database again.
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).getLast();
 | 
				
			||||||
 | 
					        assertEquals("modified content", s.getContentSectionCell());
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                2, IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void editInvalidSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("editInvalidSectionCell : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () ->
 | 
				
			||||||
 | 
					                        entrepreneurApiService.editSectionCell(
 | 
				
			||||||
 | 
					                                -1L, "should not be modified", "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).getLast();
 | 
				
			||||||
 | 
					        assertEquals("contenu très intéressant", s.getContentSectionCell());
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                1, IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void editSectionCellInvalidAccess() {
 | 
				
			||||||
 | 
					        System.out.println("editSectionCellInvalidAccess : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () ->
 | 
				
			||||||
 | 
					                        entrepreneurApiService.editSectionCell(
 | 
				
			||||||
 | 
					                                IterableToList(sectionCells3).getFirst().getIdSectionCell(),
 | 
				
			||||||
 | 
					                                "should not be modified",
 | 
				
			||||||
 | 
					                                "testentrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).getFirst();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals("contenu très intéressant", s.getContentSectionCell());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void editNullSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("editNullSectionCell : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () ->
 | 
				
			||||||
 | 
					                        entrepreneurApiService.editSectionCell(
 | 
				
			||||||
 | 
					                                null, "modified content", "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void removeValidSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("removeValidSectionCell : ");
 | 
				
			||||||
 | 
					        SectionCell tmpCell =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        new SectionCell(
 | 
				
			||||||
 | 
					                                null, 2L, "contenu temporaire", LocalDateTime.now(), project));
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                2, IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).size());
 | 
				
			||||||
 | 
					        assertDoesNotThrow(
 | 
				
			||||||
 | 
					                () ->
 | 
				
			||||||
 | 
					                        entrepreneurApiService.removeSectionCell(
 | 
				
			||||||
 | 
					                                tmpCell.getIdSectionCell(), "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                tmpCell.getIdReference(),
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, -1L))
 | 
				
			||||||
 | 
					                        .getLast()
 | 
				
			||||||
 | 
					                        .getIdReference());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void removeInvalidSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("removeInvalidSectionCell : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.removeSectionCell(-1L, "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).getFirst();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals("contenu très intéressant", s.getContentSectionCell());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void removeNullSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("removeNullSectionCell : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.removeSectionCell(null, "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void addValidSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("addValidSectionCell : ");
 | 
				
			||||||
 | 
					        SectionCell added =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        new SectionCell(null, 2L, "contenu ajouté", LocalDateTime.now(), project));
 | 
				
			||||||
 | 
					        added.updateListAnnotation(
 | 
				
			||||||
 | 
					                annotationService.addNewAnnotation(new Annotation(null, "oui j'annote")));
 | 
				
			||||||
 | 
					        added.updateAppointmentSectionCell(
 | 
				
			||||||
 | 
					                appointmentService.addNewAppointment(
 | 
				
			||||||
 | 
					                        new Appointment(
 | 
				
			||||||
 | 
					                                null,
 | 
				
			||||||
 | 
					                                LocalDate.now(),
 | 
				
			||||||
 | 
					                                LocalTime.now(),
 | 
				
			||||||
 | 
					                                LocalTime.of(2, 5),
 | 
				
			||||||
 | 
					                                "TD15",
 | 
				
			||||||
 | 
					                                "clément qui se plaint")));
 | 
				
			||||||
 | 
					        entrepreneurApiService.addSectionCell(added, "entrepreneur@mail.fr");
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).getLast();
 | 
				
			||||||
 | 
					        assertEquals("contenu ajouté", s.getContentSectionCell());
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                2, IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void addSectionCellInvalidAccess() {
 | 
				
			||||||
 | 
					        System.out.println("addSectionCellInvalidAccess : ");
 | 
				
			||||||
 | 
					        SectionCell added =
 | 
				
			||||||
 | 
					                new SectionCell(null, 2L, "contenu ajouté", LocalDateTime.now(), project);
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.addSectionCell(added, "fauxentrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					        SectionCell s =
 | 
				
			||||||
 | 
					                IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).getLast();
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                1, IterableToList(sectionCellService.getSectionCellsByProject(project, 2L)).size());
 | 
				
			||||||
 | 
					        assertEquals("contenu très intéressant", s.getContentSectionCell());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void addInvalidSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("addInvalidSectionCell : ");
 | 
				
			||||||
 | 
					        SectionCell added =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        new SectionCell(null, -1L, "contenu ajouté", LocalDateTime.now(), project));
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.addSectionCell(added, "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void addNullSectionCell() {
 | 
				
			||||||
 | 
					        System.out.println("addNullSectionCell : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.addSectionCell(null, "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void requestValidProject() {
 | 
				
			||||||
 | 
					        System.out.println("requestValidProject : ");
 | 
				
			||||||
 | 
					        int nb_project = IterableToList(this.projectService.getAllProjects()).size();
 | 
				
			||||||
 | 
					        Project validProject =
 | 
				
			||||||
 | 
					                new Project("validProject", null, LocalDate.now(), ACTIVE, null, entrepreneur);
 | 
				
			||||||
 | 
					        validProject.updateListEntrepreneurParticipation(
 | 
				
			||||||
 | 
					                IterableToList(entrepreneurService.getAllEntrepreneurs()).getLast());
 | 
				
			||||||
 | 
					        validProject.updateListSectionCell((IterableToList(sectionCells2).getFirst()));
 | 
				
			||||||
 | 
					        entrepreneurApiService.requestNewProject(validProject, "entrepreneur@mail.fr");
 | 
				
			||||||
 | 
					        assertEquals(PENDING, validProject.getProjectStatus());
 | 
				
			||||||
 | 
					        assertEquals((nb_project + 1), IterableToList(this.projectService.getAllProjects()).size());
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                IterableToList(entrepreneurService.getAllEntrepreneurs()).getLast(),
 | 
				
			||||||
 | 
					                validProject.getListEntrepreneurParticipation().getLast());
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                IterableToList(sectionCells2).getFirst().getIdSectionCell(),
 | 
				
			||||||
 | 
					                validProject.getListSectionCell().getFirst().getIdSectionCell());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void requestNullProject() {
 | 
				
			||||||
 | 
					        System.out.println("requestNullProject : ");
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.requestNewProject(null, "entrepreneur@mail.fr"));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void createNewValidAccount() {
 | 
				
			||||||
 | 
					        System.out.println("createNewValidAccount : ");
 | 
				
			||||||
 | 
					        int nb_entrepreneur = IterableToList(this.entrepreneurService.getAllEntrepreneurs()).size();
 | 
				
			||||||
 | 
					        Entrepreneur newEntrepreneur =
 | 
				
			||||||
 | 
					                new Entrepreneur(
 | 
				
			||||||
 | 
					                        "New",
 | 
				
			||||||
 | 
					                        "Test",
 | 
				
			||||||
 | 
					                        "mailtest@test.fr",
 | 
				
			||||||
 | 
					                        "mailtest2@test.fr",
 | 
				
			||||||
 | 
					                        "0888888888",
 | 
				
			||||||
 | 
					                        "ENSEIRB",
 | 
				
			||||||
 | 
					                        "ELEC",
 | 
				
			||||||
 | 
					                        false,
 | 
				
			||||||
 | 
					                        true);
 | 
				
			||||||
 | 
					        assertDoesNotThrow(() -> entrepreneurApiService.createAccount(newEntrepreneur));
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                (nb_entrepreneur + 1),
 | 
				
			||||||
 | 
					                IterableToList(this.entrepreneurService.getAllEntrepreneurs()).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void createExistingAccount() {
 | 
				
			||||||
 | 
					        System.out.println("createExistingAccount : ");
 | 
				
			||||||
 | 
					        int nb_entrepreneur = IterableToList(this.entrepreneurService.getAllEntrepreneurs()).size();
 | 
				
			||||||
 | 
					        assertThrows(
 | 
				
			||||||
 | 
					                ResponseStatusException.class,
 | 
				
			||||||
 | 
					                () -> entrepreneurApiService.createAccount(entrepreneur));
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                nb_entrepreneur,
 | 
				
			||||||
 | 
					                IterableToList(this.entrepreneurService.getAllEntrepreneurs()).size());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -0,0 +1,922 @@
 | 
				
			|||||||
 | 
					package enseirb.myinpulse;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import static enseirb.myinpulse.model.ProjectDecisionValue.*;
 | 
				
			||||||
 | 
					import static org.junit.jupiter.api.Assertions.*;
 | 
				
			||||||
 | 
					import static org.mockito.ArgumentMatchers.*;
 | 
				
			||||||
 | 
					import static org.mockito.Mockito.when;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import enseirb.myinpulse.model.*;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.SharedApiService;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.database.*;
 | 
				
			||||||
 | 
					import jakarta.persistence.EntityManager;
 | 
				
			||||||
 | 
					import jakarta.persistence.PersistenceContext;
 | 
				
			||||||
 | 
					import enseirb.myinpulse.service.UtilsService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.junit.jupiter.api.BeforeAll; // Use BeforeAll for static setup
 | 
				
			||||||
 | 
					import org.junit.jupiter.api.BeforeEach;
 | 
				
			||||||
 | 
					import org.junit.jupiter.api.Test; // Keep this import
 | 
				
			||||||
 | 
					import org.mockito.MockitoAnnotations;
 | 
				
			||||||
 | 
					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 org.springframework.http.HttpStatus;
 | 
				
			||||||
 | 
					import org.springframework.test.context.bean.override.mockito.MockitoBean;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.time.LocalDate;
 | 
				
			||||||
 | 
					import java.time.LocalDateTime;
 | 
				
			||||||
 | 
					import java.time.LocalTime;
 | 
				
			||||||
 | 
					import java.time.format.DateTimeFormatter;
 | 
				
			||||||
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					import java.util.Optional;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Helper to easily convert Iterable to List
 | 
				
			||||||
 | 
					class TestUtils {
 | 
				
			||||||
 | 
					    static <T> List<T> toList(Iterable<T> iterable) {
 | 
				
			||||||
 | 
					        List<T> list = new ArrayList<>();
 | 
				
			||||||
 | 
					        if (iterable != null) {
 | 
				
			||||||
 | 
					            iterable.forEach(list::add);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return list;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@SpringBootTest
 | 
				
			||||||
 | 
					@Transactional // Each @Test method runs in a transaction that is rolled back
 | 
				
			||||||
 | 
					public class SharedApiServiceTest {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Autowired private SharedApiService sharedApiService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Autowire actual services to use in setup and test verification
 | 
				
			||||||
 | 
					    @Autowired private ProjectService projectService;
 | 
				
			||||||
 | 
					    @Autowired private AdministratorService administratorService;
 | 
				
			||||||
 | 
					    @Autowired private EntrepreneurService entrepreneurService;
 | 
				
			||||||
 | 
					    @Autowired private SectionCellService sectionCellService;
 | 
				
			||||||
 | 
					    @Autowired private AppointmentService appointmentService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Mock UtilsService to control authorization logic
 | 
				
			||||||
 | 
					    @MockitoBean private UtilsService mockUtilsService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Static variables for data created once before all tests
 | 
				
			||||||
 | 
					    private static Project staticAuthorizedProject;
 | 
				
			||||||
 | 
					    private static String staticAuthorizedMail;
 | 
				
			||||||
 | 
					    private static Administrator staticAuthorizedAdmin;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static Project staticUnauthorizedProject;
 | 
				
			||||||
 | 
					    private static String staticUnauthorizedMail;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // --- Static Setup (Runs once before all tests) ---
 | 
				
			||||||
 | 
					    // Use @BeforeAll static method with injected services
 | 
				
			||||||
 | 
					    @BeforeAll
 | 
				
			||||||
 | 
					    static void setupOnce(
 | 
				
			||||||
 | 
					            @Autowired AdministratorService administratorService,
 | 
				
			||||||
 | 
					            @Autowired ProjectService projectService,
 | 
				
			||||||
 | 
					            @Autowired EntrepreneurService entrepreneurService) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Create and Save core test data here using injected services
 | 
				
			||||||
 | 
					        staticAuthorizedAdmin =
 | 
				
			||||||
 | 
					                administratorService.addAdministrator(getTestAdmin("static_authorized_admin"));
 | 
				
			||||||
 | 
					        staticAuthorizedMail = staticAuthorizedAdmin.getPrimaryMail();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        staticUnauthorizedProject =
 | 
				
			||||||
 | 
					                projectService.addNewProject(
 | 
				
			||||||
 | 
					                        getTestProject(
 | 
				
			||||||
 | 
					                                "static_unauthorized_project",
 | 
				
			||||||
 | 
					                                administratorService.addAdministrator(
 | 
				
			||||||
 | 
					                                        getTestAdmin("static_unauthorized_admin"))));
 | 
				
			||||||
 | 
					        staticUnauthorizedMail =
 | 
				
			||||||
 | 
					                administratorService
 | 
				
			||||||
 | 
					                        .addAdministrator(getTestAdmin("static_unauthorized_user"))
 | 
				
			||||||
 | 
					                        .getPrimaryMail(); // User who is NOT admin of the unauthorized project
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        staticAuthorizedProject =
 | 
				
			||||||
 | 
					                projectService.addNewProject(
 | 
				
			||||||
 | 
					                        getTestProject("static_authorized_project", staticAuthorizedAdmin));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Link a static entrepreneur to the authorized project if needed for some tests
 | 
				
			||||||
 | 
					        // Entrepreneur staticLinkedEntrepreneur =
 | 
				
			||||||
 | 
					        // entrepreneurService.addEntrepreneur(getTestEntrepreneur("static_linked_entrepreneur"));
 | 
				
			||||||
 | 
					        // staticAuthorizedProject.updateListEntrepreneurParticipation(staticLinkedEntrepreneur);
 | 
				
			||||||
 | 
					        // projectService.addNewProject(staticAuthorizedProject); // Re-save the project after
 | 
				
			||||||
 | 
					        // updating lists
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // --- Per-Test Setup (Runs before each test method) ---
 | 
				
			||||||
 | 
					    @BeforeEach
 | 
				
			||||||
 | 
					    void setupForEach() {
 | 
				
			||||||
 | 
					        // Reset mock expectations before each test
 | 
				
			||||||
 | 
					        MockitoAnnotations.openMocks(
 | 
				
			||||||
 | 
					                this); // Needed for mocks if not using @ExtendWith(MockitoExtension.class)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- Configure the mock UtilsService based on the actual authorization rules ---
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Rule: Any admin can check any project.
 | 
				
			||||||
 | 
					        // Assuming staticAuthorizedMail is an admin:
 | 
				
			||||||
 | 
					        when(mockUtilsService.isAllowedToCheckProject(eq(staticAuthorizedMail), anyLong()))
 | 
				
			||||||
 | 
					                .thenReturn(true); // Admin allowed for ANY project ID
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Rule: An entrepreneur can only check their own stuff.
 | 
				
			||||||
 | 
					        // Assuming staticUnauthorizedMail is an entrepreneur NOT linked to staticAuthorizedProject
 | 
				
			||||||
 | 
					        // or staticUnauthorizedProject:
 | 
				
			||||||
 | 
					        when(mockUtilsService.isAllowedToCheckProject(eq(staticUnauthorizedMail), anyLong()))
 | 
				
			||||||
 | 
					                .thenReturn(false); // Unauthorized entrepreneur NOT allowed for ANY project ID by
 | 
				
			||||||
 | 
					        // default
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // --- Helper Methods (Can remain non-static or static as needed) ---
 | 
				
			||||||
 | 
					    private static Administrator getTestAdmin(String name) {
 | 
				
			||||||
 | 
					        return new Administrator(
 | 
				
			||||||
 | 
					                name + "_surname",
 | 
				
			||||||
 | 
					                name,
 | 
				
			||||||
 | 
					                name + "@example.com",
 | 
				
			||||||
 | 
					                "secondary_" + name + "@example.com",
 | 
				
			||||||
 | 
					                "0123456789");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static Entrepreneur getTestEntrepreneur(String name) {
 | 
				
			||||||
 | 
					        return new Entrepreneur(
 | 
				
			||||||
 | 
					                name + "_surname",
 | 
				
			||||||
 | 
					                name,
 | 
				
			||||||
 | 
					                name + "@example.com",
 | 
				
			||||||
 | 
					                "secondary_" + name + "@example.com",
 | 
				
			||||||
 | 
					                "0123456789",
 | 
				
			||||||
 | 
					                "Test School",
 | 
				
			||||||
 | 
					                "Test Course",
 | 
				
			||||||
 | 
					                false);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static Project getTestProject(String name, Administrator admin) {
 | 
				
			||||||
 | 
					        Project project = new Project(name, null, LocalDate.now(), ACTIVE, admin);
 | 
				
			||||||
 | 
					        return project;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static SectionCell getTestSectionCell(
 | 
				
			||||||
 | 
					            Project project, Long sectionId, String content, LocalDateTime date) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = new SectionCell();
 | 
				
			||||||
 | 
					        sectionCell.setProjectSectionCell(project);
 | 
				
			||||||
 | 
					        sectionCell.setSectionId(sectionId);
 | 
				
			||||||
 | 
					        sectionCell.setContentSectionCell(content);
 | 
				
			||||||
 | 
					        sectionCell.setModificationDate(date);
 | 
				
			||||||
 | 
					        return sectionCell;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static SectionCell getTestSectionCell(
 | 
				
			||||||
 | 
					            Project project, Long sectionId, String content, LocalDateTime date, Long refrenceId) {
 | 
				
			||||||
 | 
					        SectionCell sectionCell = new SectionCell();
 | 
				
			||||||
 | 
					        sectionCell.setProjectSectionCell(project);
 | 
				
			||||||
 | 
					        sectionCell.setSectionId(sectionId);
 | 
				
			||||||
 | 
					        sectionCell.setContentSectionCell(content);
 | 
				
			||||||
 | 
					        sectionCell.setModificationDate(date);
 | 
				
			||||||
 | 
					        sectionCell.setIdReference(refrenceId);
 | 
				
			||||||
 | 
					        return sectionCell;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static Appointment getTestAppointment(
 | 
				
			||||||
 | 
					            LocalDate date,
 | 
				
			||||||
 | 
					            LocalTime time,
 | 
				
			||||||
 | 
					            LocalTime duration,
 | 
				
			||||||
 | 
					            String place,
 | 
				
			||||||
 | 
					            String subject,
 | 
				
			||||||
 | 
					            List<SectionCell> sectionCells,
 | 
				
			||||||
 | 
					            Report report) {
 | 
				
			||||||
 | 
					        Appointment appointment = new Appointment();
 | 
				
			||||||
 | 
					        appointment.setAppointmentDate(date);
 | 
				
			||||||
 | 
					        appointment.setAppointmentTime(time);
 | 
				
			||||||
 | 
					        appointment.setAppointmentDuration(duration);
 | 
				
			||||||
 | 
					        appointment.setAppointmentPlace(place);
 | 
				
			||||||
 | 
					        appointment.setAppointmentSubject(subject);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (sectionCells != null) {
 | 
				
			||||||
 | 
					            sectionCells.forEach(appointment::updateListSectionCell);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if (report != null) {
 | 
				
			||||||
 | 
					            appointment.setAppointmentReport(report);
 | 
				
			||||||
 | 
					            report.setAppointmentReport(appointment);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return appointment;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static Report getTestReport(String content) {
 | 
				
			||||||
 | 
					        Report report = new Report();
 | 
				
			||||||
 | 
					        report.setReportContent(content);
 | 
				
			||||||
 | 
					        return report;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     *        _____         _   ____            _   _              ____     _ _
 | 
				
			||||||
 | 
					     *       |_   _|__  ___| |_/ ___|  ___  ___| |_(_) ___  _ __  / ___|___| | |
 | 
				
			||||||
 | 
					     *         | |/ _ \/ __| __\___ \ / _ \/ __| __| |/ _ \| '_ \| |   / _ \ | |
 | 
				
			||||||
 | 
					     *         | |  __/\__ \ |_ ___) |  __/ (__| |_| | (_) | | | | |__|  __/ | |
 | 
				
			||||||
 | 
					     *         |_|\___||___/\__|____/ \___|\___|\__|_|\___/|_| |_|\____\___|_|_|
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving section cells for a specific project and section ID before a given date
 | 
				
			||||||
 | 
					     * when the user is authorized but no matching cells exist.
 | 
				
			||||||
 | 
					     * Verifies that an empty list is returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetSectionCells_Authorized_NotFound() {
 | 
				
			||||||
 | 
					        // Arrange: No specific section cells needed for this test, rely on clean @BeforeEach state
 | 
				
			||||||
 | 
					        Long targetSectionId = 1L;
 | 
				
			||||||
 | 
					        LocalDateTime dateFilter = LocalDateTime.now().plusDays(1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        Iterable<SectionCell> result =
 | 
				
			||||||
 | 
					                sharedApiService.getSectionCells(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(),
 | 
				
			||||||
 | 
					                        targetSectionId,
 | 
				
			||||||
 | 
					                        dateFilter.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm")),
 | 
				
			||||||
 | 
					                        staticAuthorizedMail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<SectionCell> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        assertTrue(resultList.isEmpty());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving section cells when the user is not authorized for the project.
 | 
				
			||||||
 | 
					     * Verifies that an Unauthorized ResponseStatusException is thrown.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetSectionCells_Unauthorized() {
 | 
				
			||||||
 | 
					        // Arrange: mockUtilsService configured in BeforeEach
 | 
				
			||||||
 | 
					        // Act & Assert
 | 
				
			||||||
 | 
					        ResponseStatusException exception =
 | 
				
			||||||
 | 
					                assertThrows(
 | 
				
			||||||
 | 
					                        ResponseStatusException.class,
 | 
				
			||||||
 | 
					                        () -> {
 | 
				
			||||||
 | 
					                            sharedApiService.getSectionCells(
 | 
				
			||||||
 | 
					                                    staticAuthorizedProject
 | 
				
			||||||
 | 
					                                            .getIdProject(), // Project static user is not
 | 
				
			||||||
 | 
					                                    // authorized for
 | 
				
			||||||
 | 
					                                    1L,
 | 
				
			||||||
 | 
					                                    LocalDateTime.now()
 | 
				
			||||||
 | 
					                                            .format(
 | 
				
			||||||
 | 
					                                                    DateTimeFormatter.ofPattern(
 | 
				
			||||||
 | 
					                                                            "yyyy-MM-dd HH:mm")),
 | 
				
			||||||
 | 
					                                    staticUnauthorizedMail); // Static unauthorized user mail
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(HttpStatus.UNAUTHORIZED, exception.getStatusCode());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving all section cells for a project when the user is authorized
 | 
				
			||||||
 | 
					     * but the project has no section cells.
 | 
				
			||||||
 | 
					     * Verifies that an empty list is returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAllSectionCells_Authorized_NoCells() {
 | 
				
			||||||
 | 
					        // Arrange: staticAuthorizedProject has no section cells initially in BeforeAll
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        Iterable<SectionCell> result =
 | 
				
			||||||
 | 
					                sharedApiService.getAllSectionCells(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), staticAuthorizedMail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<SectionCell> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        assertTrue(resultList.isEmpty());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving all section cells when the user is not authorized for the project.
 | 
				
			||||||
 | 
					     * Verifies that an Unauthorized ResponseStatusException is thrown.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAllSectionCells_Unauthorized() {
 | 
				
			||||||
 | 
					        // Arrange: mockUtilsService configured in BeforeEach
 | 
				
			||||||
 | 
					        // Act & Assert
 | 
				
			||||||
 | 
					        ResponseStatusException exception =
 | 
				
			||||||
 | 
					                assertThrows(
 | 
				
			||||||
 | 
					                        ResponseStatusException.class,
 | 
				
			||||||
 | 
					                        () -> {
 | 
				
			||||||
 | 
					                            sharedApiService.getAllSectionCells(
 | 
				
			||||||
 | 
					                                    staticAuthorizedProject.getIdProject(), staticUnauthorizedMail);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(HttpStatus.UNAUTHORIZED, exception.getStatusCode());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving section cells for a specific project and section ID before a given date
 | 
				
			||||||
 | 
					     * when the user is authorized and matching cells exist.
 | 
				
			||||||
 | 
					     * Verifies that only the correct cells are returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    // Commenting out failing test
 | 
				
			||||||
 | 
					    void testGetSectionCells_Authorized_Found() {
 | 
				
			||||||
 | 
					        Long targetSectionId = 1L;
 | 
				
			||||||
 | 
					        // Set a date filter slightly in the future so our "latest before" cell is included
 | 
				
			||||||
 | 
					        LocalDateTime dateFilter = LocalDateTime.now().plusMinutes(5);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Creating versions of the SAME SectionCell (share the same idReference)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // the first version. This will get a GENERATED idReference.
 | 
				
			||||||
 | 
					        SectionCell firstVersion =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject,
 | 
				
			||||||
 | 
					                        targetSectionId,
 | 
				
			||||||
 | 
					                        "Content V1 (Oldest)",
 | 
				
			||||||
 | 
					                        LocalDateTime.now().minusDays(3) // Oldest date
 | 
				
			||||||
 | 
					                        );
 | 
				
			||||||
 | 
					        sectionCellService.addNewSectionCell(firstVersion);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Long sharedIdReference = firstVersion.getIdReference();
 | 
				
			||||||
 | 
					        assertNotNull(
 | 
				
			||||||
 | 
					                sharedIdReference,
 | 
				
			||||||
 | 
					                "idReference should be generated after saving the first version");
 | 
				
			||||||
 | 
					        System.out.println("Generated sharedIdReference: " + sharedIdReference);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Create subsequent versions and MANUALLY set the SAME idReference.
 | 
				
			||||||
 | 
					        // These represent updates to the cell identified by sharedIdReference.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        SectionCell middleVersion =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject,
 | 
				
			||||||
 | 
					                        targetSectionId,
 | 
				
			||||||
 | 
					                        "Content V2 (Middle)",
 | 
				
			||||||
 | 
					                        LocalDateTime.now().minusDays(2), // Middle date, before filter
 | 
				
			||||||
 | 
					                        sharedIdReference);
 | 
				
			||||||
 | 
					        middleVersion = sectionCellService.addNewSectionCell(middleVersion);
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(
 | 
				
			||||||
 | 
					                middleVersion.getIdSectionCell(), sharedIdReference);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        SectionCell latestBeforeFilter =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject,
 | 
				
			||||||
 | 
					                        targetSectionId,
 | 
				
			||||||
 | 
					                        "Content V3 (Latest Before Filter)",
 | 
				
			||||||
 | 
					                        LocalDateTime.now().minusDays(1), // Latest date before filter
 | 
				
			||||||
 | 
					                        sharedIdReference);
 | 
				
			||||||
 | 
					        latestBeforeFilter = sectionCellService.addNewSectionCell(latestBeforeFilter);
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(
 | 
				
			||||||
 | 
					                latestBeforeFilter.getIdSectionCell(), sharedIdReference);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        SectionCell futureVersion =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject,
 | 
				
			||||||
 | 
					                        targetSectionId,
 | 
				
			||||||
 | 
					                        "Content V4 (Future - Should Be Excluded)",
 | 
				
			||||||
 | 
					                        LocalDateTime.now().plusDays(1), // Date is AFTER the filter
 | 
				
			||||||
 | 
					                        sharedIdReference);
 | 
				
			||||||
 | 
					        futureVersion = sectionCellService.addNewSectionCell(futureVersion);
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(
 | 
				
			||||||
 | 
					                futureVersion.getIdSectionCell(), sharedIdReference);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- Create other SectionCells that should NOT be included (different sectionId or
 | 
				
			||||||
 | 
					        // project) ---
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Cell in a different section ID
 | 
				
			||||||
 | 
					        sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject,
 | 
				
			||||||
 | 
					                        99L, // Different sectionId
 | 
				
			||||||
 | 
					                        "Content in Different Section",
 | 
				
			||||||
 | 
					                        LocalDateTime.now()));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        Iterable<SectionCell> result =
 | 
				
			||||||
 | 
					                sharedApiService.getSectionCells(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), // Use static project ID
 | 
				
			||||||
 | 
					                        targetSectionId,
 | 
				
			||||||
 | 
					                        dateFilter.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm")),
 | 
				
			||||||
 | 
					                        staticAuthorizedMail); // Use static authorized mail
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<SectionCell> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(1, resultList.size());
 | 
				
			||||||
 | 
					        // Verify that the returned cell is the 'latestBeforeFilter' cell
 | 
				
			||||||
 | 
					        // Comparing by idSectionCell is a good way to verify the exact entity
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                latestBeforeFilter.getIdSectionCell(),
 | 
				
			||||||
 | 
					                resultList.get(0).getIdSectionCell(),
 | 
				
			||||||
 | 
					                "The returned SectionCell should be the one with the latest modification date before the filter.");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Also assert the idReference and content
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                sharedIdReference,
 | 
				
			||||||
 | 
					                resultList.get(0).getIdReference(),
 | 
				
			||||||
 | 
					                "The returned cell should have the shared idReference.");
 | 
				
			||||||
 | 
					        assertEquals(
 | 
				
			||||||
 | 
					                "Content V3 (Latest Before Filter)",
 | 
				
			||||||
 | 
					                resultList.get(0).getContentSectionCell(),
 | 
				
			||||||
 | 
					                "The returned cell should have the correct content.");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving the most recent section cell for each unique idReference
 | 
				
			||||||
 | 
					     * within a project when the user is authorized and cells exist.
 | 
				
			||||||
 | 
					     * Verifies that only the latest version of each referenced cell is returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    // Tests getAllSectionCells
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    // Commenting out failing test - Removed this comment as we are fixing it
 | 
				
			||||||
 | 
					    void testGetAllSectionCells_Authorized_FoundLatest() {
 | 
				
			||||||
 | 
					        // Arrange: Create specific SectionCells for this test
 | 
				
			||||||
 | 
					        // Define the idReference values we will use for grouping
 | 
				
			||||||
 | 
					        Long refIdGroup1 = 101L;
 | 
				
			||||||
 | 
					        Long refIdGroup2 = 102L;
 | 
				
			||||||
 | 
					        Long refIdOtherProject = 103L;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- Create and Add Cells for Group 1 (refIdGroup1) ---
 | 
				
			||||||
 | 
					        // Create the older cell for group 1
 | 
				
			||||||
 | 
					        SectionCell tempOldCell1 =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject, // Project
 | 
				
			||||||
 | 
					                        1L, // Section ID (assuming this groups by section within refId)
 | 
				
			||||||
 | 
					                        "Ref1 Old", // Name
 | 
				
			||||||
 | 
					                        LocalDateTime.now().minusDays(3), // Date (older)
 | 
				
			||||||
 | 
					                        null); // Pass null or let getTestSectionCell handle it, we'll set
 | 
				
			||||||
 | 
					        // idReference later
 | 
				
			||||||
 | 
					        final SectionCell oldCell1 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(tempOldCell1); // Add to DB
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Create the newer cell for group 1
 | 
				
			||||||
 | 
					        SectionCell tempNewerCell1 =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject, // Project
 | 
				
			||||||
 | 
					                        1L, // Section ID
 | 
				
			||||||
 | 
					                        "Ref1 Newer", // Name
 | 
				
			||||||
 | 
					                        LocalDateTime.now().minusDays(2), // Date (newer than oldCell1)
 | 
				
			||||||
 | 
					                        null); // Pass null
 | 
				
			||||||
 | 
					        final SectionCell newerCell1 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(tempNewerCell1); // Add to DB
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Now, update the idReference for both cells in Group 1 to the desired value
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(oldCell1.getIdSectionCell(), refIdGroup1);
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(newerCell1.getIdSectionCell(), refIdGroup1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- Create and Add Cells for Group 2 (refIdGroup2) ---
 | 
				
			||||||
 | 
					        // Create the older cell for group 2
 | 
				
			||||||
 | 
					        SectionCell tempOldCell2 =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject, // Project
 | 
				
			||||||
 | 
					                        2L, // Section ID (different section)
 | 
				
			||||||
 | 
					                        "Ref2 Old", // Name
 | 
				
			||||||
 | 
					                        LocalDateTime.now().minusDays(1), // Date (older than newerCell2)
 | 
				
			||||||
 | 
					                        null); // Pass null
 | 
				
			||||||
 | 
					        final SectionCell oldCell2 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(tempOldCell2); // Add to DB
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Create the newer cell for group 2
 | 
				
			||||||
 | 
					        SectionCell tempNewerCell2 =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject, // Project
 | 
				
			||||||
 | 
					                        2L, // Section ID
 | 
				
			||||||
 | 
					                        "Ref2 Newer", // Name
 | 
				
			||||||
 | 
					                        LocalDateTime.now(), // Date (latest)
 | 
				
			||||||
 | 
					                        null); // Pass null
 | 
				
			||||||
 | 
					        final SectionCell newerCell2 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(tempNewerCell2); // Add to DB
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Now, update the idReference for both cells in Group 2 to the desired value
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(oldCell2.getIdSectionCell(), refIdGroup2);
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(newerCell2.getIdSectionCell(), refIdGroup2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- Create and Add Cell for Other Project (refIdOtherProject) ---
 | 
				
			||||||
 | 
					        Project otherProject =
 | 
				
			||||||
 | 
					                projectService.addNewProject(
 | 
				
			||||||
 | 
					                        getTestProject(
 | 
				
			||||||
 | 
					                                "other_project_for_cell_test",
 | 
				
			||||||
 | 
					                                administratorService.addAdministrator(
 | 
				
			||||||
 | 
					                                        getTestAdmin("other_admin_cell_test"))));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        SectionCell tempOtherProjectCell =
 | 
				
			||||||
 | 
					                getTestSectionCell(
 | 
				
			||||||
 | 
					                        otherProject, // DIFFERENT Project
 | 
				
			||||||
 | 
					                        1L, // Section ID
 | 
				
			||||||
 | 
					                        "Other Project Cell", // Name
 | 
				
			||||||
 | 
					                        LocalDateTime.now(), // Date
 | 
				
			||||||
 | 
					                        null); // Pass null
 | 
				
			||||||
 | 
					        final SectionCell otherProjectCell =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(tempOtherProjectCell); // Add to DB
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Now, update the idReference for the Other Project cell
 | 
				
			||||||
 | 
					        sectionCellService.updateSectionCellReferenceId(
 | 
				
			||||||
 | 
					                otherProjectCell.getIdSectionCell(), refIdOtherProject);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        // Ensure the service call uses the correct project ID and mail
 | 
				
			||||||
 | 
					        Iterable<SectionCell> result =
 | 
				
			||||||
 | 
					                sharedApiService.getAllSectionCells(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), // Use static project ID
 | 
				
			||||||
 | 
					                        staticAuthorizedMail); // Use static authorized mail
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<SectionCell> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        // We expect 2 cells from the staticAuthorizedProject:
 | 
				
			||||||
 | 
					        // - The latest one from refIdGroup1 (newerCell1)
 | 
				
			||||||
 | 
					        // - The latest one from refIdGroup2 (newerCell2)
 | 
				
			||||||
 | 
					        assertEquals(2, resultList.size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert that the result list contains the LATEST cell from each group within the correct
 | 
				
			||||||
 | 
					        // project
 | 
				
			||||||
 | 
					        assertTrue(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                cell ->
 | 
				
			||||||
 | 
					                                        cell.getIdSectionCell()
 | 
				
			||||||
 | 
					                                                .equals(newerCell1.getIdSectionCell())),
 | 
				
			||||||
 | 
					                "Should contain the latest cell for Group 1"); // Add assertion message
 | 
				
			||||||
 | 
					        assertTrue(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                cell ->
 | 
				
			||||||
 | 
					                                        cell.getIdSectionCell()
 | 
				
			||||||
 | 
					                                                .equals(newerCell2.getIdSectionCell())),
 | 
				
			||||||
 | 
					                "Should contain the latest cell for Group 2"); // Add assertion message
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert that the result list does NOT contain the OLDER cells from the correct project
 | 
				
			||||||
 | 
					        assertFalse(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                cell ->
 | 
				
			||||||
 | 
					                                        cell.getIdSectionCell()
 | 
				
			||||||
 | 
					                                                .equals(oldCell1.getIdSectionCell())),
 | 
				
			||||||
 | 
					                "Should not contain the older cell for Group 1"); // Add assertion message
 | 
				
			||||||
 | 
					        assertFalse(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                cell ->
 | 
				
			||||||
 | 
					                                        cell.getIdSectionCell()
 | 
				
			||||||
 | 
					                                                .equals(oldCell2.getIdSectionCell())),
 | 
				
			||||||
 | 
					                "Should not contain the older cell for Group 2"); // Add assertion message
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert that the result list does NOT contain the cell from the other project
 | 
				
			||||||
 | 
					        assertFalse(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                cell ->
 | 
				
			||||||
 | 
					                                        cell.getIdSectionCell()
 | 
				
			||||||
 | 
					                                                .equals(otherProjectCell.getIdSectionCell())),
 | 
				
			||||||
 | 
					                "Should not contain cells from other projects"); // Add assertion message
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     *        _____         _    ____      _   ____            _           _   ____
 | 
				
			||||||
 | 
					     *       |_   _|__  ___| |_ / ___| ___| |_|  _ \ _ __ ___ (_) ___  ___| |_| __ ) _   _
 | 
				
			||||||
 | 
					     *         | |/ _ \/ __| __| |  _ / _ \ __| |_) | '__/ _ \| |/ _ \/ __| __|  _ \| | | |
 | 
				
			||||||
 | 
					     *         | |  __/\__ \ |_| |_| |  __/ |_|  __/| | | (_) | |  __/ (__| |_| |_) | |_| |
 | 
				
			||||||
 | 
					     *        _|_|\___||___/\__|\____|\___|\__|_|   |_|  \___// |\___|\___|\__|____/ \__, |
 | 
				
			||||||
 | 
					     *       |_ _|  _ \                                     |__/                     |___/
 | 
				
			||||||
 | 
					     *        | || | | |
 | 
				
			||||||
 | 
					     *        | || |_| |
 | 
				
			||||||
 | 
					     *       |___|____/
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving entrepreneurs linked to a project when the user is authorized
 | 
				
			||||||
 | 
					     * but no entrepreneurs are linked.
 | 
				
			||||||
 | 
					     * Verifies that an empty list is returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetEntrepreneursByProjectId_Authorized_NotFound() {
 | 
				
			||||||
 | 
					        // Arrange: staticAuthorizedProject has no entrepreneurs linked initially in BeforeAll
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        Iterable<Entrepreneur> result =
 | 
				
			||||||
 | 
					                sharedApiService.getEntrepreneursByProjectId(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), staticAuthorizedMail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<Entrepreneur> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        assertTrue(resultList.isEmpty());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving entrepreneurs linked to a project when the user is not authorized.
 | 
				
			||||||
 | 
					     * Verifies that an Unauthorized ResponseStatusException is thrown.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetEntrepreneursByProjectId_Unauthorized() {
 | 
				
			||||||
 | 
					        // Arrange: mockUtilsService configured in BeforeEach
 | 
				
			||||||
 | 
					        // Act & Assert
 | 
				
			||||||
 | 
					        ResponseStatusException exception =
 | 
				
			||||||
 | 
					                assertThrows(
 | 
				
			||||||
 | 
					                        ResponseStatusException.class,
 | 
				
			||||||
 | 
					                        () -> {
 | 
				
			||||||
 | 
					                            sharedApiService.getEntrepreneursByProjectId(
 | 
				
			||||||
 | 
					                                    staticAuthorizedProject.getIdProject(), staticUnauthorizedMail);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(HttpStatus.UNAUTHORIZED, exception.getStatusCode());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     *    _____         _    ____      _      _       _           _       ____
 | 
				
			||||||
 | 
					     *   |_   _|__  ___| |_ / ___| ___| |_   / \   __| |_ __ ___ (_)_ __ | __ ) _   _
 | 
				
			||||||
 | 
					     *     | |/ _ \/ __| __| |  _ / _ \ __| / _ \ / _` | '_ ` _ \| | '_ \|  _ \| | | |
 | 
				
			||||||
 | 
					     *     | |  __/\__ \ |_| |_| |  __/ |_ / ___ \ (_| | | | | | | | | | | |_) | |_| |
 | 
				
			||||||
 | 
					     *    _|_|\___||___/\__|\____|\___|\__/_/   \_\__,_|_| |_| |_|_|_| |_|____/ \__, |
 | 
				
			||||||
 | 
					     *   |_ _|  _ \                                                             |___/
 | 
				
			||||||
 | 
					     *    | || | | |
 | 
				
			||||||
 | 
					     *    | || |_| |
 | 
				
			||||||
 | 
					     *   |___|____/
 | 
				
			||||||
 | 
					     *
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving appointments linked to a project's section cells when the user is authorized
 | 
				
			||||||
 | 
					     * but no such appointments exist.
 | 
				
			||||||
 | 
					     * Verifies that an empty list is returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAppointmentsByProjectId_Authorized_NotFound() {
 | 
				
			||||||
 | 
					        // Arrange: staticAuthorizedProject has no linked section cells or appointments initially
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        Iterable<Appointment> result =
 | 
				
			||||||
 | 
					                sharedApiService.getAppointmentsByProjectId(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), staticAuthorizedMail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<Appointment> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        assertTrue(resultList.isEmpty());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving the administrator linked to a project when the user is authorized
 | 
				
			||||||
 | 
					     * and an administrator is linked.
 | 
				
			||||||
 | 
					     * Verifies that the correct administrator is returned.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    // Tests getAdminByProjectId
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAdminByProjectId_Authorized_Found() {
 | 
				
			||||||
 | 
					        // Arrange: staticAuthorizedProject is created with staticAuthorizedAdmin in BeforeAll
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        Administrator result =
 | 
				
			||||||
 | 
					                sharedApiService.getAdminByProjectId(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), staticAuthorizedMail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        assertNotNull(result);
 | 
				
			||||||
 | 
					        assertEquals(staticAuthorizedAdmin.getIdUser(), result.getIdUser());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving the administrator linked to a project when the user is not authorized.
 | 
				
			||||||
 | 
					     * Verifies that an Unauthorized ResponseStatusException is thrown.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAdminByProjectId_Unauthorized() {
 | 
				
			||||||
 | 
					        // Arrange: mockUtilsService configured in BeforeEach
 | 
				
			||||||
 | 
					        // Act & Assert
 | 
				
			||||||
 | 
					        ResponseStatusException exception =
 | 
				
			||||||
 | 
					                assertThrows(
 | 
				
			||||||
 | 
					                        ResponseStatusException.class,
 | 
				
			||||||
 | 
					                        () -> {
 | 
				
			||||||
 | 
					                            sharedApiService.getAdminByProjectId(
 | 
				
			||||||
 | 
					                                    staticAuthorizedProject.getIdProject(), staticUnauthorizedMail);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(HttpStatus.UNAUTHORIZED, exception.getStatusCode());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     *    _____         _
 | 
				
			||||||
 | 
					     *   |_   _|__  ___| |_
 | 
				
			||||||
 | 
					     *     | |/ _ \/ __| __|
 | 
				
			||||||
 | 
					     *     | |  __/\__ \ |_
 | 
				
			||||||
 | 
					     *     |_|\___||___/\__|        _       _                            _
 | 
				
			||||||
 | 
					     *      / \   _ __  _ __   ___ (_)_ __ | |_ ___ _ __ ___   ___ _ __ | |_ ___
 | 
				
			||||||
 | 
					     *     / _ \ | '_ \| '_ \ / _ \| | '_ \| __/ _ \ '_ ` _ \ / _ \ '_ \| __/ __|
 | 
				
			||||||
 | 
					     *    / ___ \| |_) | |_) | (_) | | | | | ||  __/ | | | | |  __/ | | | |_\__ \
 | 
				
			||||||
 | 
					     *   /_/   \_\ .__/| .__/ \___/|_|_| |_|\__\___|_| |_| |_|\___|_| |_|\__|___/
 | 
				
			||||||
 | 
					     *           |_|   |_|
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests retrieving appointments linked to a project's section cells when the user is not authorized.
 | 
				
			||||||
 | 
					     * Verifies that an Unauthorized ResponseStatusException is thrown.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAppointmentsByProjectId_Unauthorized() {
 | 
				
			||||||
 | 
					        // Arrange: mockUtilsService configured in BeforeEach
 | 
				
			||||||
 | 
					        // Act & Assert
 | 
				
			||||||
 | 
					        ResponseStatusException exception =
 | 
				
			||||||
 | 
					                assertThrows(
 | 
				
			||||||
 | 
					                        ResponseStatusException.class,
 | 
				
			||||||
 | 
					                        () -> {
 | 
				
			||||||
 | 
					                            sharedApiService.getAppointmentsByProjectId(
 | 
				
			||||||
 | 
					                                    staticAuthorizedProject.getIdProject(), staticUnauthorizedMail);
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertEquals(HttpStatus.UNAUTHORIZED, exception.getStatusCode());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @PersistenceContext // Inject EntityManager
 | 
				
			||||||
 | 
					    private EntityManager entityManager;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Assume these static variables are defined elsewhere in your test class
 | 
				
			||||||
 | 
					    // private static Project staticAuthorizedProject;
 | 
				
			||||||
 | 
					    // private static String staticAuthorizedMail;
 | 
				
			||||||
 | 
					    // private static Administrator staticAuthorizedAdmin;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Assume getTestSectionCell, getTestProject, getTestAdmin, getTestAppointment, TestUtils.toList
 | 
				
			||||||
 | 
					    // are defined elsewhere
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    void testGetAppointmentsByProjectId_Authorized_Found() {
 | 
				
			||||||
 | 
					        // Arrange: Create specific SectionCells and Appointments for this test
 | 
				
			||||||
 | 
					        SectionCell cell1 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        getTestSectionCell(
 | 
				
			||||||
 | 
					                                staticAuthorizedProject, 1L, "Cell 1 Test", LocalDateTime.now()));
 | 
				
			||||||
 | 
					        SectionCell cell2 =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        getTestSectionCell(
 | 
				
			||||||
 | 
					                                staticAuthorizedProject, 2L, "Cell 2 Test", LocalDateTime.now()));
 | 
				
			||||||
 | 
					        Project otherProject =
 | 
				
			||||||
 | 
					                projectService.addNewProject(
 | 
				
			||||||
 | 
					                        getTestProject(
 | 
				
			||||||
 | 
					                                "other_project_app_test",
 | 
				
			||||||
 | 
					                                administratorService.addAdministrator(
 | 
				
			||||||
 | 
					                                        getTestAdmin("other_admin_app_test"))));
 | 
				
			||||||
 | 
					        SectionCell otherProjectCell =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        getTestSectionCell(
 | 
				
			||||||
 | 
					                                otherProject,
 | 
				
			||||||
 | 
					                                1L,
 | 
				
			||||||
 | 
					                                "Other Project Cell App Test",
 | 
				
			||||||
 | 
					                                LocalDateTime.now()));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Create Appointments with SectionCells lists (Owning side)
 | 
				
			||||||
 | 
					        Appointment app1 =
 | 
				
			||||||
 | 
					                getTestAppointment(
 | 
				
			||||||
 | 
					                        LocalDate.now().plusDays(10),
 | 
				
			||||||
 | 
					                        LocalTime.NOON,
 | 
				
			||||||
 | 
					                        LocalTime.of(0, 30),
 | 
				
			||||||
 | 
					                        "Place 1 App Test",
 | 
				
			||||||
 | 
					                        "Subject 1 App Test",
 | 
				
			||||||
 | 
					                        List.of(cell1), // This links Appointment to SectionCell
 | 
				
			||||||
 | 
					                        null);
 | 
				
			||||||
 | 
					        Appointment savedApp1 = appointmentService.addNewAppointment(app1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Appointment app2 =
 | 
				
			||||||
 | 
					                getTestAppointment(
 | 
				
			||||||
 | 
					                        LocalDate.now().plusDays(11),
 | 
				
			||||||
 | 
					                        LocalTime.NOON.plusHours(1),
 | 
				
			||||||
 | 
					                        LocalTime.of(1, 0),
 | 
				
			||||||
 | 
					                        "Place 2 App Test",
 | 
				
			||||||
 | 
					                        "Subject 2 App Test",
 | 
				
			||||||
 | 
					                        List.of(cell1, cell2), // This links Appointment to SectionCells
 | 
				
			||||||
 | 
					                        null);
 | 
				
			||||||
 | 
					        Appointment savedApp2 = appointmentService.addNewAppointment(app2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Appointment otherApp =
 | 
				
			||||||
 | 
					                getTestAppointment(
 | 
				
			||||||
 | 
					                        LocalDate.now().plusDays(12),
 | 
				
			||||||
 | 
					                        LocalTime.MIDNIGHT,
 | 
				
			||||||
 | 
					                        LocalTime.of(0, 15),
 | 
				
			||||||
 | 
					                        "Other Place App Test",
 | 
				
			||||||
 | 
					                        "Other Subject App Test",
 | 
				
			||||||
 | 
					                        List.of(otherProjectCell), // This links Appointment to SectionCell
 | 
				
			||||||
 | 
					                        null);
 | 
				
			||||||
 | 
					        Appointment savedOtherApp =
 | 
				
			||||||
 | 
					                appointmentService.addNewAppointment(otherApp); // Capture saved entity
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- IMPORTANT DEBUGGING STEPS ---
 | 
				
			||||||
 | 
					        // Flush pending changes to the database (including join table inserts)
 | 
				
			||||||
 | 
					        entityManager.flush();
 | 
				
			||||||
 | 
					        // Clear the persistence context cache to ensure entities are loaded fresh from the database
 | 
				
			||||||
 | 
					        entityManager.clear();
 | 
				
			||||||
 | 
					        // --- END IMPORTANT DEBUGGING STEPS ---
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // --- Add Debug Logging Here ---
 | 
				
			||||||
 | 
					        // Re-fetch cells to see their state after saving Appointments and flushing/clearing cache
 | 
				
			||||||
 | 
					        // These fetches should load from the database due to entityManager.clear()
 | 
				
			||||||
 | 
					        SectionCell fetchedCell1_postPersist =
 | 
				
			||||||
 | 
					                sectionCellService.getSectionCellById(cell1.getIdSectionCell());
 | 
				
			||||||
 | 
					        SectionCell fetchedCell2_postPersist =
 | 
				
			||||||
 | 
					                sectionCellService.getSectionCellById(cell2.getIdSectionCell());
 | 
				
			||||||
 | 
					        SectionCell fetchedOtherCell_postPersist =
 | 
				
			||||||
 | 
					                sectionCellService.getSectionCellById(otherProjectCell.getIdSectionCell());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Access the lazy collections to see if they are populated from the DB
 | 
				
			||||||
 | 
					        // This access should trigger lazy loading if the data is in the DB
 | 
				
			||||||
 | 
					        List<Appointment> cell1Apps_postPersist =
 | 
				
			||||||
 | 
					                fetchedCell1_postPersist.getAppointmentSectionCell();
 | 
				
			||||||
 | 
					        List<Appointment> cell2Apps_postPersist =
 | 
				
			||||||
 | 
					                fetchedCell2_postPersist.getAppointmentSectionCell();
 | 
				
			||||||
 | 
					        List<Appointment> otherCellApps_postPersist =
 | 
				
			||||||
 | 
					                fetchedOtherCell_postPersist.getAppointmentSectionCell();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Ensure logging is enabled in SharedApiService and SectionCellService methods called below
 | 
				
			||||||
 | 
					        Iterable<Appointment> result =
 | 
				
			||||||
 | 
					                sharedApiService.getAppointmentsByProjectId(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject.getIdProject(), // Use static project ID
 | 
				
			||||||
 | 
					                        staticAuthorizedMail); // Use static authorized mail
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<Appointment> resultList = TestUtils.toList(result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert
 | 
				
			||||||
 | 
					        assertEquals(2, resultList.size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertTrue(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(a -> a.getIdAppointment().equals(savedApp1.getIdAppointment())));
 | 
				
			||||||
 | 
					        assertTrue(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(a -> a.getIdAppointment().equals(savedApp2.getIdAppointment())));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertFalse(
 | 
				
			||||||
 | 
					                resultList.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                a ->
 | 
				
			||||||
 | 
					                                        a.getIdAppointment()
 | 
				
			||||||
 | 
					                                                .equals(savedOtherApp.getIdAppointment())));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Tests creating a new appointment request when the user is authorized
 | 
				
			||||||
 | 
					     * for the project linked to the appointment's section cell.
 | 
				
			||||||
 | 
					     * Verifies that the appointment and its relationships are saved correctly in the database.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    // Tests createAppointmentRequest
 | 
				
			||||||
 | 
					    @Test
 | 
				
			||||||
 | 
					    // Commenting out failing test
 | 
				
			||||||
 | 
					    void testCreateAppointmentRequest_Authorized_Success() {
 | 
				
			||||||
 | 
					        // Arrange: Create transient appointment linked to a cell in the static authorized project
 | 
				
			||||||
 | 
					        LocalDate date = LocalDate.parse("2026-01-01");
 | 
				
			||||||
 | 
					        LocalTime time = LocalTime.parse("10:00:00");
 | 
				
			||||||
 | 
					        LocalTime duration = LocalTime.parse("00:30:00");
 | 
				
			||||||
 | 
					        String place = "Meeting Room Integrated";
 | 
				
			||||||
 | 
					        String subject = "Discuss Project Integrated";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        SectionCell linkedCell =
 | 
				
			||||||
 | 
					                sectionCellService.addNewSectionCell(
 | 
				
			||||||
 | 
					                        getTestSectionCell(
 | 
				
			||||||
 | 
					                                staticAuthorizedProject,
 | 
				
			||||||
 | 
					                                0L,
 | 
				
			||||||
 | 
					                                "Related Section Content Integrated",
 | 
				
			||||||
 | 
					                                LocalDateTime.now()));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Report newReport = null; // getTestReport(reportContent); // Uses no-arg constructor
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Appointment newAppointment =
 | 
				
			||||||
 | 
					                getTestAppointment(
 | 
				
			||||||
 | 
					                        date, time, duration, place, subject, List.of(linkedCell), newReport);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // mockUtilsService is configured in BeforeEach to allow staticAuthorizedMail for
 | 
				
			||||||
 | 
					        // staticAuthorizedProject
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Act
 | 
				
			||||||
 | 
					        // Allow the service method to call the actual appointmentService.addNewAppointment
 | 
				
			||||||
 | 
					        assertDoesNotThrow(
 | 
				
			||||||
 | 
					                () ->
 | 
				
			||||||
 | 
					                        sharedApiService.createAppointmentRequest(
 | 
				
			||||||
 | 
					                                newAppointment, staticAuthorizedMail));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Assert: Retrieve the appointment from the DB and verify it and its relationships were
 | 
				
			||||||
 | 
					        // saved
 | 
				
			||||||
 | 
					        // We find it by looking for appointments linked to the authorized project's cells
 | 
				
			||||||
 | 
					        Iterable<SectionCell> projectCells =
 | 
				
			||||||
 | 
					                sectionCellService.getSectionCellsByProject(
 | 
				
			||||||
 | 
					                        staticAuthorizedProject, linkedCell.getSectionId()); // Fetch relevant cells
 | 
				
			||||||
 | 
					        List<Appointment> projectAppointmentsList = new ArrayList<>();
 | 
				
			||||||
 | 
					        projectCells.forEach(
 | 
				
			||||||
 | 
					                cell ->
 | 
				
			||||||
 | 
					                        projectAppointmentsList.addAll(
 | 
				
			||||||
 | 
					                                sectionCellService.getAppointmentsBySectionCellId(
 | 
				
			||||||
 | 
					                                        cell.getIdSectionCell()))); // Get appointments for
 | 
				
			||||||
 | 
					        // those cells
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Optional<Appointment> createdAppointmentOpt =
 | 
				
			||||||
 | 
					                projectAppointmentsList.stream()
 | 
				
			||||||
 | 
					                        .filter(
 | 
				
			||||||
 | 
					                                a ->
 | 
				
			||||||
 | 
					                                        a.getAppointmentDate().equals(date)
 | 
				
			||||||
 | 
					                                                && a.getAppointmentTime().equals(time)
 | 
				
			||||||
 | 
					                                                && a.getAppointmentPlace().equals(place)
 | 
				
			||||||
 | 
					                                                && a.getAppointmentSubject().equals(subject))
 | 
				
			||||||
 | 
					                        .findFirst();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        assertTrue(createdAppointmentOpt.isPresent());
 | 
				
			||||||
 | 
					        Appointment createdAppointment = createdAppointmentOpt.get();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // FIX: Corrected bidirectional link check
 | 
				
			||||||
 | 
					        assertEquals(1, createdAppointment.getAppointmentListSectionCell().size());
 | 
				
			||||||
 | 
					        assertTrue(
 | 
				
			||||||
 | 
					                createdAppointment.getAppointmentListSectionCell().stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                sc -> sc.getIdSectionCell().equals(linkedCell.getIdSectionCell())));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        List<Appointment> appointmentsLinkedToCell =
 | 
				
			||||||
 | 
					                TestUtils.toList(
 | 
				
			||||||
 | 
					                        sectionCellService.getAppointmentsBySectionCellId(
 | 
				
			||||||
 | 
					                                linkedCell.getIdSectionCell()));
 | 
				
			||||||
 | 
					        assertTrue(
 | 
				
			||||||
 | 
					                appointmentsLinkedToCell.stream()
 | 
				
			||||||
 | 
					                        .anyMatch(
 | 
				
			||||||
 | 
					                                a ->
 | 
				
			||||||
 | 
					                                        a.getIdAppointment()
 | 
				
			||||||
 | 
					                                                .equals(createdAppointment.getIdAppointment())));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										22
									
								
								config/.env.dev
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								config/.env.dev
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
				
			|||||||
 | 
					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/
 | 
				
			||||||
@@ -16,7 +16,7 @@ BACKEND_PASSWORD=backend_db_user_password
 | 
				
			|||||||
DATABASE_URL=localhost:5433
 | 
					DATABASE_URL=localhost:5433
 | 
				
			||||||
 | 
					
 | 
				
			||||||
VITE_KEYCLOAK_URL=http://localhost:7080
 | 
					VITE_KEYCLOAK_URL=http://localhost:7080
 | 
				
			||||||
VITE_KEYCLOAK_CLIENT_ID=myinpulse-dev
 | 
					VITE_KEYCLOAK_CLIENT_ID=MyINPulse-vite
 | 
				
			||||||
VITE_KEYCLOAK_REALM=test
 | 
					VITE_KEYCLOAK_REALM=MyINPulse
 | 
				
			||||||
VITE_APP_URL=http://localhost:5173
 | 
					VITE_APP_URL=http://localhost:5173
 | 
				
			||||||
VITE_BACKEND_URL=http://localhost:8081/
 | 
					VITE_BACKEND_URL=http://localhost:8081/
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										12
									
								
								documentation/Doc.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								documentation/Doc.txt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
				
			|||||||
 | 
					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
 | 
				
			||||||
							
								
								
									
										13
									
								
								documentation/openapi/notes.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								documentation/openapi/notes.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,13 @@
 | 
				
			|||||||
 | 
					## API Endpoints notes
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### EntrepreneurApi and SharedApi
 | 
				
			||||||
 | 
					#### Endpoint Name Changes
 | 
				
			||||||
 | 
					- `/entrepreneur/lcsection/modify/{sectionId}` → `/entrepreneur/sectionCell/modify/{sectionId}`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Admin api
 | 
				
			||||||
 | 
					- `/admin/appointments/report/{appointmentId}` has no PUT and DELETE
 | 
				
			||||||
 | 
					- `/admin/request-join` and `/admin/request-join/decision/{joinRequestId}` have not yet been implemented
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Unauth api
 | 
				
			||||||
 | 
					- `/unauth/request-join/{projectId}` has not yet been implemented
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										11
									
								
								documentation/openapi/run_doc.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										11
									
								
								documentation/openapi/run_doc.sh
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,11 @@
 | 
				
			|||||||
 | 
					#!/bin/bash
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					cd ./swagger-ui
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if [ ! -d "./node_modules/" ]
 | 
				
			||||||
 | 
					then
 | 
				
			||||||
 | 
					    npm install
 | 
				
			||||||
 | 
					    npm install swagger-cli
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					npm start
 | 
				
			||||||
							
								
								
									
										387
									
								
								documentation/openapi/src/adminApi.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										387
									
								
								documentation/openapi/src/adminApi.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,387 @@
 | 
				
			|||||||
 | 
					# Admin API Endpoints
 | 
				
			||||||
 | 
					paths:
 | 
				
			||||||
 | 
					  /admin/projects:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getAdminProjects
 | 
				
			||||||
 | 
					      summary: Get projects associated with the admin
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves a list of projects managed by the requesting admin, including details for overview.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of projects returned successfully.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/project"
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid project data provided (e.g., missing required fields).
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized - Authentication required or invalid token.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: addProjectManually
 | 
				
			||||||
 | 
					      summary: Manually add a new project
 | 
				
			||||||
 | 
					      description: Creates a new project with the provided details. (NOTE that this meant for manually inserting projects, for example importing already existing projects).
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Project details to create. `idProject` and `creationDate` will be ignored if sent and set by the server.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/project"
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200": # Use 200 Created for successful creation
 | 
				
			||||||
 | 
					          description: Created - Project added successfully. Returns the created project.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                $ref: "./main.yaml#/components/schemas/project"
 | 
				
			||||||
 | 
					        "409":
 | 
				
			||||||
 | 
					          description: Bad Request - Project already exists.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/projects/pending:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getPendingProjects
 | 
				
			||||||
 | 
					      summary: Get projects awaiting validation
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves a list of projects submitted by entrepreneurs that are pending admin approval.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of pending projects returned.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/project" # Assuming pending projects use the same schema
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.      
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/request-join:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getPendingProjects
 | 
				
			||||||
 | 
					      summary: Get entrepreneurs project join requests
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      description:  Retrieves a list of pending requests from entrepreneurs to join an existing project.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of pending project join requests.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/joinRequest"
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					  
 | 
				
			||||||
 | 
					  /admin/request-join/decision/{joinRequestId}:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      summary: Approve or reject a pending project join request
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: joinRequestId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the pending join request to decide upon.
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					      description: |-
 | 
				
			||||||
 | 
					        Allows an admin to make a decision on an ebtrepreneur's request to join an existing project awaiting validation.
 | 
				
			||||||
 | 
					        If approved (isAccepted=true), the entrepreneur is linked to the project with ID joinRequestId.
 | 
				
			||||||
 | 
					        If rejected (isAccepted=false), the pending request data might be archived or deleted based on business logic.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - No Content, decision processed successfully..
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/joinRequestDecision"
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input (e.g., missing decision).
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.   
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/projects/pending/decision:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: decidePendingProject
 | 
				
			||||||
 | 
					      summary: Approve or reject a pending project
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      description: |-
 | 
				
			||||||
 | 
					        Allows an admin to make a decision on a project awaiting validation.
 | 
				
			||||||
 | 
					        If approved (isAccepted=true), the project status changes, and it's linked to the involved users.
 | 
				
			||||||
 | 
					        If rejected (isAccepted=false), the pending project data might be archived or deleted based on business logic.
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					          - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Decision payload.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: './main.yaml#/components/schemas/projectDecision'
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200": # Use 200 No Content for successful action with no body
 | 
				
			||||||
 | 
					          description: No Content - Decision processed successfully.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input (e.g., missing decision).
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/pending-accounts: # Path updated
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getPendingAccounts
 | 
				
			||||||
 | 
					      summary: Get accounts awaiting validation
 | 
				
			||||||
 | 
					      description: Retrieves a list of entrepreneur user accounts that are pending admin validation.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					          - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of pending accounts returned.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/user-entrepreneur"
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/accounts/validate/{userId}:
 | 
				
			||||||
 | 
					    post: # Changed to POST as it changes state
 | 
				
			||||||
 | 
					      operationId: validateUserAccount
 | 
				
			||||||
 | 
					      summary: Validate a pending user account
 | 
				
			||||||
 | 
					      description: Marks the user account specified by userId as validated/active.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					          - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: userId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the user account to validate.
 | 
				
			||||||
 | 
					          example: 102
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: No Content - Account validated successfully.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid user ID format.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.  
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/appointments/upcoming:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getUpcomingAppointments
 | 
				
			||||||
 | 
					      summary: Get upcoming appointments for an admin
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves a list of appointments scheduled for an admin in the future.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of upcoming appointments.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/appointment"
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: no appointments found.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/appointments/report/{appointmentId}:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: createAppointmentReport
 | 
				
			||||||
 | 
					      summary: Create a report for an appointment
 | 
				
			||||||
 | 
					      description: Creates and links a new report (e.g., meeting minutes) to the specified appointment using the provided content.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: appointmentId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: ID of the appointment to add a report to.
 | 
				
			||||||
 | 
					          example: 303
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Report content. `idReport` will be ignored if sent.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/report"
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Created - Report created and linked successfully. Returns the created report.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					             application/json:
 | 
				
			||||||
 | 
					                schema: { $ref: "./main.yaml#/components/schemas/report" }
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input (e.g., missing content, invalid appointment ID format).
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					    put: # Changed to PUT for update/replacement
 | 
				
			||||||
 | 
					      operationId: updateAppointmentReport
 | 
				
			||||||
 | 
					      summary: Update an existing appointment report
 | 
				
			||||||
 | 
					      description: Updates the content of an existing report linked to the specified appointment. Replaces the entire report content.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: appointmentId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: ID of the appointment whose report needs updating.
 | 
				
			||||||
 | 
					          example: 303
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: New report content. `idReport` in the body should match the existing report's ID or will be ignored.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/report"
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - Report updated successfully. Returns the updated report.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					             application/json:
 | 
				
			||||||
 | 
					                schema: { $ref: "./main.yaml#/components/schemas/report" }
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input (e.g., missing content).
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/projects/{projectId}:
 | 
				
			||||||
 | 
					    delete:
 | 
				
			||||||
 | 
					      operationId: removeProject
 | 
				
			||||||
 | 
					      summary: Remove a project
 | 
				
			||||||
 | 
					      description: Permanently removes the project specified by projectId and potentially related data (use with caution).
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					          - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: projectId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the project to remove.
 | 
				
			||||||
 | 
					          example: 12
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: No Content - Project removed successfully.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid project ID format.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /admin/make-admin/{userId}:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: grantAdminRights
 | 
				
			||||||
 | 
					      summary: Grant admin rights to a user
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Elevates the specified user to also have administrator privileges. Assumes the user already exists.
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: userId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the user to grant admin rights.
 | 
				
			||||||
 | 
					          example: 103
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200": # Use 200 No Content
 | 
				
			||||||
 | 
					          description: No Content - Admin rights granted successfully.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid user ID format or user is already an admin.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					  /admin/create-account:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      summary: Creates Admin out Jwt Token
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Admin API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Create an admin instance in the MyINPulse DB of the information provided from the authenticated user's keycloack token.
 | 
				
			||||||
 | 
					       The information required in the token are `userSurname`, `username`, `primaryMail`, `secondaryMail`, `phoneNumber`.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200": 
 | 
				
			||||||
 | 
					          description: No Content - Admin user created successfully.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
							
								
								
									
										197
									
								
								documentation/openapi/src/entrepreneurApi.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										197
									
								
								documentation/openapi/src/entrepreneurApi.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,197 @@
 | 
				
			|||||||
 | 
					# Entrepreneur API Endpoints
 | 
				
			||||||
 | 
					paths:
 | 
				
			||||||
 | 
					  /entrepreneur/projects/request:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: requestProjectCreation
 | 
				
			||||||
 | 
					      summary: Request creation and validation of a new project
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Entrepreneurs API
 | 
				
			||||||
 | 
					      description: |-
 | 
				
			||||||
 | 
					        Submits a request for a new project. The project details are provided in the request body.
 | 
				
			||||||
 | 
					        The requesting entrepreneur (identified by the token) will be associated to it.
 | 
				
			||||||
 | 
					        The project is created with a 'pending' status, awaiting admin approval.
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Project details for the request. `status`, `creationDate` are required by the model when being sent but is ignored by the server; 
 | 
				
			||||||
 | 
					          primarily expects a valid `projectId`, `name`, `logo`.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/project"
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Accepted - Project creation request received and is pending validation.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input (e.g., missing name).
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					  /entrepreneur/sectionCells: # Base path
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: addSectionCell
 | 
				
			||||||
 | 
					      summary: Add a cell to a Lean Canvas section
 | 
				
			||||||
 | 
					      description: Adds a new cell (like a sticky note) with the provided content to a specific section of the entrepreneur's project's Lean Canvas. Assumes project context is known based on user's token.
 | 
				
			||||||
 | 
					        `idSectionCell` and `modificationDate` are server-generated so they're values in the request are ignored by the server.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Entrepreneurs API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Section cell details. `idSectionCell` and `modificationDate` will be ignored if sent.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/sectionCell"
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Created - Section cell added successfully. Returns the created cell.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input (e.g., missing content or sectionId).
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					  /entrepreneur/sectionCells/{sectionCellId}:
 | 
				
			||||||
 | 
					    put:
 | 
				
			||||||
 | 
					      operationId: modifySectionCell
 | 
				
			||||||
 | 
					      summary: Modify data in a Lean Canvas section cell
 | 
				
			||||||
 | 
					      description: Updates the content of an existing Lean Canvas section cell specified by `sectionCellId`. The server "updates" (it keeps a record of the previous version to keep a history of all the sectionCells and creates a new ones with the specified modifications) the `modificationDate`.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Entrepreneurs API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: sectionCellId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the section cell to modify.
 | 
				
			||||||
 | 
					          example: 508
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Updated section cell details. `sectionCellId` "the path parameter" is the only id that's consideredn the `sectionCellId` id in the request body is ignored. `modificationDate` should be updated by the server.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/sectionCell"
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - Section cell updated successfully. Returns the updated cell.
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input or ID mismatch.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    delete:
 | 
				
			||||||
 | 
					      operationId: removeSectionCell
 | 
				
			||||||
 | 
					      summary: Remove a Lean Canvas section cell
 | 
				
			||||||
 | 
					      description: Deletes the Lean Canvas section cell specified by `sectionCellId`.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Entrepreneurs API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: sectionCellId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the section cell to remove.
 | 
				
			||||||
 | 
					          example: 509
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: No Content - Section cell removed successfully.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid ID format.
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Bad Request - sectionCell not found.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /entrepreneur/projects:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      summary: gets the projectId of the project associated with the entrepreneur 
 | 
				
			||||||
 | 
					      description: returns a list of projectIds of the projects associated with the entrepreneur
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Entrepreneurs API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - Section cell updated successfully. Returns the updated cell.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/project"
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input or ID mismatch.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized or identity not found
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /entrepreneur/projects/project-is-active:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      summary: checks if the project associated with an entrepreneur is active 
 | 
				
			||||||
 | 
					      description: returns a boolean if the project associated with an entrepreneur has an active status 
 | 
				
			||||||
 | 
					        (i.e has been validated by an admin). The user should be routed to LeanCanvas. any other response code 
 | 
				
			||||||
 | 
					        should be treated as false
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Entrepreneurs API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - got the value successfully any other response code should be treated as false.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: boolean
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input or ID mismatch.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized or identity not found
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /entrepreneur/projects/has-pending-request:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      summary: checks if the user has a pending projectRequest 
 | 
				
			||||||
 | 
					      description: returns a boolean if the project associated with an entrepreneur has a pending status 
 | 
				
			||||||
 | 
					        (i.e has not yet been validated by an admin). The user should be routed to a page telling him that he should 
 | 
				
			||||||
 | 
					        wait for admin validation. any other response code should be treated as false.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Entrepreneurs API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur]
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - got the value successfully any other response code should be treated as false.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: boolean
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid input or ID mismatch.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized or identity not found
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
							
								
								
									
										159
									
								
								documentation/openapi/src/main.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								documentation/openapi/src/main.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,159 @@
 | 
				
			|||||||
 | 
					openapi: 3.0.3
 | 
				
			||||||
 | 
					info:
 | 
				
			||||||
 | 
					  title: MyInpulse Backend API
 | 
				
			||||||
 | 
					  description: This serves as an OpenAPI documentation for the MyInpulse backend service, covering operations for Entrepreneurs, Admins, and shared functionalities.
 | 
				
			||||||
 | 
					  version: 0.2.1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					tags:
 | 
				
			||||||
 | 
					  - name: Entrepreneurs API
 | 
				
			||||||
 | 
					    description: API endpoints primarily for Entrepreneur users.
 | 
				
			||||||
 | 
					  - name: Admin API
 | 
				
			||||||
 | 
					    description: API endpoints restricted to Admin users for management tasks.
 | 
				
			||||||
 | 
					  - name: Shared API
 | 
				
			||||||
 | 
					    description: API endpoints accessible by both Entrepreneurs and Admins.
 | 
				
			||||||
 | 
					  - name: Unauth API
 | 
				
			||||||
 | 
					    description: API endpoints related to user account management.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					components:
 | 
				
			||||||
 | 
					  schemas:
 | 
				
			||||||
 | 
					    user:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/user"
 | 
				
			||||||
 | 
					    user-entrepreneur:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/user-entrepreneur"
 | 
				
			||||||
 | 
					    user-admin:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/user-admin"
 | 
				
			||||||
 | 
					    sectionCell:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/sectionCell"
 | 
				
			||||||
 | 
					    project:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/project"
 | 
				
			||||||
 | 
					    report:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/report"
 | 
				
			||||||
 | 
					    appointment:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/appointment"
 | 
				
			||||||
 | 
					    joinRequest:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/joinRequest"
 | 
				
			||||||
 | 
					    projectDecision:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/projectDecision"
 | 
				
			||||||
 | 
					    joinRequestDecision:
 | 
				
			||||||
 | 
					      $ref: "models.yaml#/joinRequestDecision"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  securitySchemes:
 | 
				
			||||||
 | 
					      MyINPulse:
 | 
				
			||||||
 | 
					        type: oauth2
 | 
				
			||||||
 | 
					        description: OAuth2 authentication using Keycloak.
 | 
				
			||||||
 | 
					        flows:
 | 
				
			||||||
 | 
					          implicit:
 | 
				
			||||||
 | 
					            authorizationUrl: '{keycloakBaseUrl}/realms/{keycloakRealm}/protocol/openid-connect/auth'
 | 
				
			||||||
 | 
					            scopes:
 | 
				
			||||||
 | 
					              MyINPulse-admin: Grants administrator access.
 | 
				
			||||||
 | 
					              MyINPulse-entrepreneur: Grants standard entrepreneur user access.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					servers:
 | 
				
			||||||
 | 
					    - url: '{serverProtocol}://{serverHost}:{serverPort}'
 | 
				
			||||||
 | 
					      description: API Server
 | 
				
			||||||
 | 
					      variables:
 | 
				
			||||||
 | 
					        serverProtocol:
 | 
				
			||||||
 | 
					          enum: [http, https]
 | 
				
			||||||
 | 
					          default: http
 | 
				
			||||||
 | 
					        serverHost:
 | 
				
			||||||
 | 
					          default: localhost
 | 
				
			||||||
 | 
					        serverPort:
 | 
				
			||||||
 | 
					          enum: ['8081'] 
 | 
				
			||||||
 | 
					          default: '8081'
 | 
				
			||||||
 | 
					        keycloakBaseUrl:
 | 
				
			||||||
 | 
					          default: http://localhost:7080
 | 
				
			||||||
 | 
					          description: Base URL for the Keycloak server.
 | 
				
			||||||
 | 
					        keycloakRealm:
 | 
				
			||||||
 | 
					          default: MyInpulseRealm 
 | 
				
			||||||
 | 
					          description: Keycloak realm name.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					paths:
 | 
				
			||||||
 | 
					  #       _   _                   _   _        _          _ 
 | 
				
			||||||
 | 
					  #      | | | |_ __   __ _ _   _| |_| |__    / \   _ __ (_)
 | 
				
			||||||
 | 
					  #      | | | | '_ \ / _` | | | | __| '_ \  / _ \ | '_ \| |
 | 
				
			||||||
 | 
					  #      | |_| | | | | (_| | |_| | |_| | | |/ ___ \| |_) | |
 | 
				
			||||||
 | 
					  #       \___/|_| |_|\__,_|\__,_|\__|_| |_/_/   \_\ .__/|_|
 | 
				
			||||||
 | 
					  #                                                |_|      
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /unauth/finalize:
 | 
				
			||||||
 | 
					    $ref: "./unauthApi.yaml#/paths/~1unauth~1finalize"
 | 
				
			||||||
 | 
					  /unauth/request-join/{projectId}:
 | 
				
			||||||
 | 
					    $ref: "./unauthApi.yaml#/paths/~1unauth~1request-join~1{projectId}"
 | 
				
			||||||
 | 
					  /unauth/request-admin-role:
 | 
				
			||||||
 | 
					    $ref: "./unauthApi.yaml#/paths/~1unauth~1request-admin-role"
 | 
				
			||||||
 | 
					  /unauth/check-if-not-pending:
 | 
				
			||||||
 | 
					    $ref: "./unauthApi.yaml#/paths/~1unauth~1check-if-not-pending"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #          _    ____  __  __ ___ _   _      _    ____ ___
 | 
				
			||||||
 | 
					  #         / \  |  _ \|  \/  |_ _| \ | |    / \  |  _ \_ _|
 | 
				
			||||||
 | 
					  #        / _ \ | | | | |\/| || ||  \| |   / _ \ | |_) | |
 | 
				
			||||||
 | 
					  #       / ___ \| |_| | |  | || || |\  |  / ___ \|  __/| |
 | 
				
			||||||
 | 
					  #      /_/   \_\____/|_|  |_|___|_| \_| /_/   \_\_|  |___|
 | 
				
			||||||
 | 
					  #
 | 
				
			||||||
 | 
					  /admin/pending-accounts:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1pending-accounts"
 | 
				
			||||||
 | 
					  /admin/accounts/validate/{userId}:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1accounts~1validate~1{userId}"
 | 
				
			||||||
 | 
					  /admin/request-join:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1request-join"
 | 
				
			||||||
 | 
					  /admin/request-join/decision/{joinRequestId}:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1request-join~1decision~1{joinRequestId}"
 | 
				
			||||||
 | 
					  /admin/projects:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1projects"
 | 
				
			||||||
 | 
					  /admin/projects/pending:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1projects~1pending"
 | 
				
			||||||
 | 
					  /admin/projects/pending/decision:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1projects~1pending~1decision"
 | 
				
			||||||
 | 
					  /admin/appointments/report/{appointmentId}:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1appointments~1report~1{appointmentId}"
 | 
				
			||||||
 | 
					  /admin/appointments/upcoming:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1appointments~1upcoming"
 | 
				
			||||||
 | 
					  /admin/projects/{projectId}:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1projects~1{projectId}" 
 | 
				
			||||||
 | 
					  /admin/make-admin/{userId}:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1make-admin~1{userId}"
 | 
				
			||||||
 | 
					  /admin/create-account:
 | 
				
			||||||
 | 
					    $ref: "./adminApi.yaml#/paths/~1admin~1create-account"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #       ____  _                        _      _    ____ ___
 | 
				
			||||||
 | 
					  #      / ___|| |__   __ _ _ __ ___  __| |    / \  |  _ \_ _|
 | 
				
			||||||
 | 
					  #      \___ \| '_ \ / _` | '__/ _ \/ _` |   / _ \ | |_) | |
 | 
				
			||||||
 | 
					  #       ___) | | | | (_| | | |  __/ (_| |  / ___ \|  __/| |
 | 
				
			||||||
 | 
					  #      |____/|_| |_|\__,_|_|  \___|\__,_| /_/   \_\_|  |___|
 | 
				
			||||||
 | 
					  #
 | 
				
			||||||
 | 
					  /shared/projects/sectionCells/{projectId}/{sectionId}/{date}:
 | 
				
			||||||
 | 
					    $ref: "./sharedApi.yaml#/paths/~1shared~1projects~1sectionCells~1{projectId}~1{sectionId}~1{date}"
 | 
				
			||||||
 | 
					  /shared/projects/entrepreneurs/{projectId}:
 | 
				
			||||||
 | 
					    $ref: "./sharedApi.yaml#/paths/~1shared~1projects~1entrepreneurs~1{projectId}"
 | 
				
			||||||
 | 
					  /shared/projects/admin/{projectId}:
 | 
				
			||||||
 | 
					    $ref: "./sharedApi.yaml#/paths/~1shared~1projects~1admin~1{projectId}"
 | 
				
			||||||
 | 
					  /shared/projects/appointments/{projectId}:
 | 
				
			||||||
 | 
					    $ref: "./sharedApi.yaml#/paths/~1shared~1projects~1appointments~1{projectId}"
 | 
				
			||||||
 | 
					  /shared/appointments/report/{appointmentId}:
 | 
				
			||||||
 | 
					    $ref: "./sharedApi.yaml#/paths/~1shared~1appointments~1report~1{appointmentId}"
 | 
				
			||||||
 | 
					  /shared/appointments/request:
 | 
				
			||||||
 | 
					    $ref: "./sharedApi.yaml#/paths/~1shared~1appointments~1request"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #      _____ _   _ _____ ____  _____ ____  ____  _____ _   _ _____ _   _ ____
 | 
				
			||||||
 | 
					  #     | ____| \ | |_   _|  _ \| ____|  _ \|  _ \| ____| \ | | ____| | | |  _ \
 | 
				
			||||||
 | 
					  #     |  _| |  \| | | | | |_) |  _| | |_) | |_) |  _| |  \| |  _| | | | | |_) |
 | 
				
			||||||
 | 
					  #     | |___| |\  | | | |  _ <| |___|  __/|  _ <| |___| |\  | |___| |_| |  _ <
 | 
				
			||||||
 | 
					  #     |_____|_|_\_| |_| |_| \_\_____|_|   |_| \_\_____|_| \_|_____|\___/|_| \_\
 | 
				
			||||||
 | 
					  #        / \  |  _ \_ _|
 | 
				
			||||||
 | 
					  #       / _ \ | |_) | |
 | 
				
			||||||
 | 
					  #      / ___ \|  __/| |
 | 
				
			||||||
 | 
					  #     /_/   \_\_|  |___|
 | 
				
			||||||
 | 
					  #
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /entrepreneur/projects:
 | 
				
			||||||
 | 
					    $ref: "./entrepreneurApi.yaml#/paths/~1entrepreneur~1projects"
 | 
				
			||||||
 | 
					  /entrepreneur/projects/request:
 | 
				
			||||||
 | 
					    $ref: "./entrepreneurApi.yaml#/paths/~1entrepreneur~1projects~1request"
 | 
				
			||||||
 | 
					  /entrepreneur/sectionCells:
 | 
				
			||||||
 | 
					    $ref: "./entrepreneurApi.yaml#/paths/~1entrepreneur~1sectionCells"
 | 
				
			||||||
 | 
					  /entrepreneur/sectionCells/{sectionCellId}:
 | 
				
			||||||
 | 
					    $ref: "./entrepreneurApi.yaml#/paths/~1entrepreneur~1sectionCells~1{sectionCellId}"
 | 
				
			||||||
 | 
					  /entrepreneur/projects/project-is-active:
 | 
				
			||||||
 | 
					    $ref: "./entrepreneurApi.yaml#/paths/~1entrepreneur~1projects~1project-is-active"
 | 
				
			||||||
 | 
					  /entrepreneur/projects/has-pending-request:
 | 
				
			||||||
 | 
					    $ref: "./entrepreneurApi.yaml#/paths/~1entrepreneur~1projects~1has-pending-request"
 | 
				
			||||||
							
								
								
									
										210
									
								
								documentation/openapi/src/models.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										210
									
								
								documentation/openapi/src/models.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,210 @@
 | 
				
			|||||||
 | 
					# models.yaml
 | 
				
			||||||
 | 
					user:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    idUser:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: Unique identifier for the user.
 | 
				
			||||||
 | 
					      #readOnly: true # Typically generated by the server
 | 
				
			||||||
 | 
					      example: 101
 | 
				
			||||||
 | 
					    userSurname:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: User's surname (last name).
 | 
				
			||||||
 | 
					      example: "Doe"
 | 
				
			||||||
 | 
					    userName:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: User's given name (first name).
 | 
				
			||||||
 | 
					      example: "John"
 | 
				
			||||||
 | 
					    primaryMail:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: email
 | 
				
			||||||
 | 
					      description: User's primary email address.
 | 
				
			||||||
 | 
					      example: "john.doe@example.com"
 | 
				
			||||||
 | 
					    secondaryMail:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: email
 | 
				
			||||||
 | 
					      description: User's secondary email address (optional).
 | 
				
			||||||
 | 
					      example: "j.doe@personal.com"
 | 
				
			||||||
 | 
					    phoneNumber:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: User's phone number.
 | 
				
			||||||
 | 
					      example: "+33612345678" # Example using international format
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					user-entrepreneur:
 | 
				
			||||||
 | 
					  allOf:
 | 
				
			||||||
 | 
					    - $ref: "#/user"
 | 
				
			||||||
 | 
					    - type: object
 | 
				
			||||||
 | 
					      properties:
 | 
				
			||||||
 | 
					        school:
 | 
				
			||||||
 | 
					          type: string
 | 
				
			||||||
 | 
					          description: The school the entrepreneur attends/attended.
 | 
				
			||||||
 | 
					          example: "ENSEIRB-MATMECA"
 | 
				
			||||||
 | 
					        course:
 | 
				
			||||||
 | 
					          type: string
 | 
				
			||||||
 | 
					          description: The specific course or program of study.
 | 
				
			||||||
 | 
					          example: "Electronics"
 | 
				
			||||||
 | 
					        sneeStatus:
 | 
				
			||||||
 | 
					          type: boolean
 | 
				
			||||||
 | 
					          description: Indicates if the user has SNEE status (Statut National d'Étudiant-Entrepreneur).
 | 
				
			||||||
 | 
					          example: true
 | 
				
			||||||
 | 
					  example: # Added full object example
 | 
				
			||||||
 | 
					    idUser: 101
 | 
				
			||||||
 | 
					    userSurname: "Doe"
 | 
				
			||||||
 | 
					    userName: "John"
 | 
				
			||||||
 | 
					    primaryMail: "john.doe@example.com"
 | 
				
			||||||
 | 
					    secondaryMail: "j.doe@personal.com"
 | 
				
			||||||
 | 
					    phoneNumber: "+33612345678"
 | 
				
			||||||
 | 
					    school: "ENSEIRB-MATMECA"
 | 
				
			||||||
 | 
					    course: "Electronics"
 | 
				
			||||||
 | 
					    sneeStatus: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					user-admin:
 | 
				
			||||||
 | 
					  allOf:
 | 
				
			||||||
 | 
					    - $ref: "#/user"
 | 
				
			||||||
 | 
					  # No additional properties needed for this example
 | 
				
			||||||
 | 
					  example: # Added full object example
 | 
				
			||||||
 | 
					    idUser: 55
 | 
				
			||||||
 | 
					    userSurname: "Admin"
 | 
				
			||||||
 | 
					    userName: "Super"
 | 
				
			||||||
 | 
					    primaryMail: "admin@myinpulse.com"
 | 
				
			||||||
 | 
					    phoneNumber: "+33512345678"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					sectionCell:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a cell (like a sticky note) within a specific section of a project's Lean Canvas.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    idSectionCell:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: Unique identifier for the section cell.
 | 
				
			||||||
 | 
					      #readOnly: true # Generated by server
 | 
				
			||||||
 | 
					      example: 508
 | 
				
			||||||
 | 
					    sectionId:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: Identifier of the Lean Canvas section this cell belongs to (e.g., 1 for Problem, 2 for Solution).
 | 
				
			||||||
 | 
					      example: 1
 | 
				
			||||||
 | 
					    contentSectionCell:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: The text content of the section cell.
 | 
				
			||||||
 | 
					      example: "Users find it hard to track project progress."
 | 
				
			||||||
 | 
					    modificationDate:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: date # Using Java LocalDate -> YYYY-MM-DD
 | 
				
			||||||
 | 
					      description: The date when this cell was last modified.
 | 
				
			||||||
 | 
					      #readOnly: true # Typically updated by the server on modification
 | 
				
			||||||
 | 
					      example: "yyyy-MM-dd HH:mm"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					project:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a project being managed or developed.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    idProject:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: Unique identifier for the project.
 | 
				
			||||||
 | 
					      #readOnly: true # Generated by server
 | 
				
			||||||
 | 
					      example: 12
 | 
				
			||||||
 | 
					    projectName:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: The name of the project.
 | 
				
			||||||
 | 
					      example: "MyInpulse Mobile App"
 | 
				
			||||||
 | 
					    creationDate:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: date # Using Java LocalDate -> YYYY-MM-DD
 | 
				
			||||||
 | 
					      description: The date when the project was created in the system.
 | 
				
			||||||
 | 
					      #readOnly: true # Set by server
 | 
				
			||||||
 | 
					      example: "yyyy-MM-dd HH:mm"
 | 
				
			||||||
 | 
					    logo:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: byte
 | 
				
			||||||
 | 
					      description: Base64 encoded string representing the project logo image.
 | 
				
			||||||
 | 
					      example: "/*Base64 encoded string representing the project logo image*/"
 | 
				
			||||||
 | 
					    status:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      enum: [PENDING, ACTIVE, ENDED, ABORTED, REJECTED]
 | 
				
			||||||
 | 
					      description: Corresponds to a status enum internal to the backend, it's value in in requests
 | 
				
			||||||
 | 
					        incoming to the server should be ignored as the client shouldn't be specifying them.
 | 
				
			||||||
 | 
					      example: "NaN"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					joinRequest:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a request from an entrepreneur to join an already existing project.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    idProject:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: the ID of the project the entrepreneur wants to join.
 | 
				
			||||||
 | 
					      example: 42
 | 
				
			||||||
 | 
					    entrepreneur:
 | 
				
			||||||
 | 
					      $ref: "#/user-entrepreneur"
 | 
				
			||||||
 | 
					      
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					report:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a report associated with an appointment.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    idReport:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: Unique identifier for the report.
 | 
				
			||||||
 | 
					      #readOnly: true # Generated by server
 | 
				
			||||||
 | 
					      example: 987
 | 
				
			||||||
 | 
					    reportContent:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: The textual content of the report. Could be plain text or Markdown (specify if known).
 | 
				
			||||||
 | 
					      example: "Discussed roadmap milestones for Q3. Agreed on preliminary UI mockups."
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					appointment: # Corrected typo
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a scheduled meeting or appointment.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    idAppointment: # Assuming there's an ID
 | 
				
			||||||
 | 
					        type: integer
 | 
				
			||||||
 | 
					        description: Unique identifier for the appointment.
 | 
				
			||||||
 | 
					        #readOnly: true
 | 
				
			||||||
 | 
					        example: 303
 | 
				
			||||||
 | 
					    appointmentDate:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: date # Using Java LocalDate -> YYYY-MM-DD
 | 
				
			||||||
 | 
					      description: The date of the appointment.
 | 
				
			||||||
 | 
					      example: "2025-05-10"
 | 
				
			||||||
 | 
					    appointmentTime:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      format: time # Using Java LocalTime -> HH:mm:ss
 | 
				
			||||||
 | 
					      description: The time of the appointment (local time).
 | 
				
			||||||
 | 
					      example: "14:30:00"
 | 
				
			||||||
 | 
					    appointmentDuration:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: Duration of the appointment in ISO 8601 duration format (e.g., PT1H30M for 1 hour 30 minutes).
 | 
				
			||||||
 | 
					      example: "PT1H" # Example for 1 hour
 | 
				
			||||||
 | 
					    appointmentPlace:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: Location or meeting link for the appointment.
 | 
				
			||||||
 | 
					      example: "Meeting Room 3 / https://meet.example.com/abc-def-ghi"
 | 
				
			||||||
 | 
					    appointmentSubject:
 | 
				
			||||||
 | 
					      type: string
 | 
				
			||||||
 | 
					      description: The main topic or subject of the appointment.
 | 
				
			||||||
 | 
					      example: "Q3 Roadmap Planning"
 | 
				
			||||||
 | 
					    # Consider adding project ID or user IDs if relevant association exists
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					projectDecision:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a decision from an admin to accept a pending project.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    projectId:
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: The ID of the project the entrepreneur wants to join.
 | 
				
			||||||
 | 
					      example: 12
 | 
				
			||||||
 | 
					    adminId: 
 | 
				
			||||||
 | 
					      type: integer
 | 
				
			||||||
 | 
					      description: The ID of the project the admin who will supervise the project in case of admission.
 | 
				
			||||||
 | 
					      example: 2
 | 
				
			||||||
 | 
					    isAccepted:
 | 
				
			||||||
 | 
					      type: boolean
 | 
				
			||||||
 | 
					      description: The boolean value of the decision.
 | 
				
			||||||
 | 
					      example: "true"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					joinRequestDecision:
 | 
				
			||||||
 | 
					  type: object
 | 
				
			||||||
 | 
					  description: Represents a decision from an admin to accept a pending project join request.
 | 
				
			||||||
 | 
					  properties:
 | 
				
			||||||
 | 
					    isAccepted:
 | 
				
			||||||
 | 
					      type: boolean
 | 
				
			||||||
 | 
					      description: The boolean value of the decision.
 | 
				
			||||||
 | 
					      example: "true"
 | 
				
			||||||
							
								
								
									
										193
									
								
								documentation/openapi/src/sharedApi.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										193
									
								
								documentation/openapi/src/sharedApi.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,193 @@
 | 
				
			|||||||
 | 
					# Shared API Endpoints
 | 
				
			||||||
 | 
					paths:    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /shared/projects/sectionCells/{projectId}/{sectionId}/{date}: 
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getSectionCellsByDate
 | 
				
			||||||
 | 
					      summary: Get project section cells modified on a specific date
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Shared API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur, MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves section cells belonging to a specific section of a project, filtered by the last modification date. Requires user to have access to the project.
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: projectId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: integer }
 | 
				
			||||||
 | 
					          description: ID of the project.
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: sectionId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: integer }
 | 
				
			||||||
 | 
					          description: ID of the Lean Canvas section.
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: date
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: string, format: date } # Expect YYYY-MM-DD
 | 
				
			||||||
 | 
					          description: The modification date to filter by (YYYY-MM-DD HH:mm).
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of section cells matching the criteria.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/sectionCell"
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid parameter format.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /shared/projects/entrepreneurs/{projectId}:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getProjectEntrepreneurs
 | 
				
			||||||
 | 
					      summary: Get entrepreneurs associated with a project
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Shared API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur, MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves a list of entrepreneur users associated with the specified project. Requires access to the project.
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: projectId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: integer }
 | 
				
			||||||
 | 
					          description: ID of the project.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of entrepreneurs.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/user-entrepreneur"
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.       
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.         
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Not Found - Project not found.       
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /shared/projects/admin/{projectId}: # Path updated
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getProjectAdmin
 | 
				
			||||||
 | 
					      summary: Get admin associated with a project
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Shared API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur, MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves a list of admin users associated with the specified project. Requires access to the project.
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: projectId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: integer }
 | 
				
			||||||
 | 
					          description: ID of the project.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - admin.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                $ref: "./main.yaml#/components/schemas/user-admin"
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.       
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.    
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Not Found - Project not found.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /shared/projects/appointments/{projectId}:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getProjectAppointments
 | 
				
			||||||
 | 
					      summary: Get appointments related to a project
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Shared API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur, MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves a list of appointments associated with the specified project. Requires access to the project.
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: projectId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: integer }
 | 
				
			||||||
 | 
					          description: ID of the project.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - List of appointments.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: array
 | 
				
			||||||
 | 
					                items:
 | 
				
			||||||
 | 
					                  $ref: "./main.yaml#/components/schemas/appointment"
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /shared/appointments/report/{appointmentId}: # Path updated
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      operationId: getAppointmentReport # Shared endpoint implies read-only access might be possible
 | 
				
			||||||
 | 
					      summary: Get the report for an appointment
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Shared API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur, MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Retrieves the report associated with a specific appointment. Requires user to have access to the appointment/project.
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: appointmentId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema: { type: integer }
 | 
				
			||||||
 | 
					          description: ID of the appointment.
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: OK - Report PDF returned.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/pdf:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                schema:
 | 
				
			||||||
 | 
					                type: string
 | 
				
			||||||
 | 
					                format: binary
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /shared/appointments/request:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      operationId: requestAppointment
 | 
				
			||||||
 | 
					      summary: Request a new appointment
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Shared API
 | 
				
			||||||
 | 
					      security:
 | 
				
			||||||
 | 
					        - MyINPulse: [MyINPulse-entrepreneur, MyINPulse-admin]
 | 
				
			||||||
 | 
					      description: Allows a user (entrepreneur or admin) to request a new appointment, potentially with another user or regarding a project. Details in the body. The request might need confirmation or create a pending appointment.
 | 
				
			||||||
 | 
					      requestBody:
 | 
				
			||||||
 | 
					        required: true
 | 
				
			||||||
 | 
					        description: Details of the appointment request.
 | 
				
			||||||
 | 
					        content:
 | 
				
			||||||
 | 
					          application/json:
 | 
				
			||||||
 | 
					            schema:
 | 
				
			||||||
 | 
					              $ref: "./main.yaml#/components/schemas/appointment" # Assuming request uses same model structure
 | 
				
			||||||
 | 
					                # Potentially add projectId or targetUserId here
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200": # Accepted seems appropriate for a request
 | 
				
			||||||
 | 
					          description: Accepted - Appointment request submitted.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid appointment details.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					          
 | 
				
			||||||
							
								
								
									
										90
									
								
								documentation/openapi/src/unauthApi.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								documentation/openapi/src/unauthApi.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,90 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					#       _   _                   _   _        _          _ 
 | 
				
			||||||
 | 
					#      | | | |_ __   __ _ _   _| |_| |__    / \   _ __ (_)
 | 
				
			||||||
 | 
					#      | | | | '_ \ / _` | | | | __| '_ \  / _ \ | '_ \| |
 | 
				
			||||||
 | 
					#      | |_| | | | | (_| | |_| | |_| | | |/ ___ \| |_) | |
 | 
				
			||||||
 | 
					#       \___/|_| |_|\__,_|\__,_|\__|_| |_/_/   \_\ .__/|_|
 | 
				
			||||||
 | 
					#                                                |_|      
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					paths:
 | 
				
			||||||
 | 
					  /unauth/finalize:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      summary: Finalize account setup using authentication token
 | 
				
			||||||
 | 
					      description: |-
 | 
				
			||||||
 | 
					        Completes the user account creation/setup process in the MyInpulse system.
 | 
				
			||||||
 | 
					        This endpoint requires the user to be authenticated via Keycloak (e.g., after initial login).
 | 
				
			||||||
 | 
					        User details (name, email, etc.) are extracted from the authenticated user's token (e.g., Keycloak JWT).
 | 
				
			||||||
 | 
					        No request body is needed. The account is marked as pending admin validation upon successful finalization.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					        - Unauth API
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Created - Account finalized and pending admin validation. Returns the user profile.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Problem processing the token or user data derived from it.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized - Valid authentication token required.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					  /unauth/request-join/{projectId}:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      summary: Request to join an existing project
 | 
				
			||||||
 | 
					      description: Submits a request for the authenticated user (keycloack authenticated) to join the project specified by projectId. Their role is then changed to entrepreneur in server and Keycloak. This requires approval from a project admin.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Unauth API
 | 
				
			||||||
 | 
					      parameters:
 | 
				
			||||||
 | 
					        - in: path
 | 
				
			||||||
 | 
					          name: projectId
 | 
				
			||||||
 | 
					          required: true
 | 
				
			||||||
 | 
					          schema:
 | 
				
			||||||
 | 
					            type: integer
 | 
				
			||||||
 | 
					          description: The ID of the project to request joining.
 | 
				
			||||||
 | 
					          example: 15
 | 
				
			||||||
 | 
					      responses: # Moved responses block to correct level
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Accepted - Join request submitted and pending approval.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid project ID format 
 | 
				
			||||||
 | 
					        "409":
 | 
				
			||||||
 | 
					          description: Already member/request pending.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					  /unauth/request-admin-role:
 | 
				
			||||||
 | 
					    post:
 | 
				
			||||||
 | 
					      summary: Request to become an admin
 | 
				
			||||||
 | 
					      description: Submits a request for the authenticated user (keycloack authenticated) to become an admin. Their role is then changed to admin in server and Keycloak. This requires approval from a project admin.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Unauth API
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Accepted - Become admin request submitted and pending approval.
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid project ID format or already member/request pending.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /unauth/check-if-not-pending:
 | 
				
			||||||
 | 
					    get:
 | 
				
			||||||
 | 
					      summary: Returns a boolean of whether the user's account is not pending
 | 
				
			||||||
 | 
					      description: Returns a boolean with value `true` if the user's account is not pending and `false` if it is.
 | 
				
			||||||
 | 
					      tags:
 | 
				
			||||||
 | 
					       - Unauth API
 | 
				
			||||||
 | 
					      responses:
 | 
				
			||||||
 | 
					        "200":
 | 
				
			||||||
 | 
					          description: Accepted - Become admin request submitted and pending approval.
 | 
				
			||||||
 | 
					          content:
 | 
				
			||||||
 | 
					            application/json:
 | 
				
			||||||
 | 
					              schema:
 | 
				
			||||||
 | 
					                type: boolean  
 | 
				
			||||||
 | 
					        "400":
 | 
				
			||||||
 | 
					          description: Bad Request - Invalid project ID format or already member/request pending.
 | 
				
			||||||
 | 
					        "401":
 | 
				
			||||||
 | 
					          description: Unauthorized.
 | 
				
			||||||
 | 
					        "404":
 | 
				
			||||||
 | 
					          description: Bad Request - User not found in database.
 | 
				
			||||||
 | 
					        "403":
 | 
				
			||||||
 | 
					          description: Bad Token - Invalid Keycloack configuration.
 | 
				
			||||||
							
								
								
									
										14
									
								
								documentation/openapi/swagger-ui/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								documentation/openapi/swagger-ui/main.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
				
			|||||||
 | 
					const express = require("express");
 | 
				
			||||||
 | 
					const swaggerUi = require("swagger-ui-express");
 | 
				
			||||||
 | 
					const yaml = require("js-yaml");
 | 
				
			||||||
 | 
					const fs = require("fs");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const app = express();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const swaggerDocument = yaml.load(fs.readFileSync("../src/bundled.yaml", "utf8"));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					app.use("/api-docs", swaggerUi.serve, swaggerUi.setup(swaggerDocument));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					app.listen(3000, () => {
 | 
				
			||||||
 | 
					    console.log("Swagger UI running at http://localhost:3000/api-docs");
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
							
								
								
									
										2178
									
								
								documentation/openapi/swagger-ui/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										2178
									
								
								documentation/openapi/swagger-ui/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										21
									
								
								documentation/openapi/swagger-ui/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								documentation/openapi/swagger-ui/package.json
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					{
 | 
				
			||||||
 | 
					  "name": "swagger-ui",
 | 
				
			||||||
 | 
					  "version": "1.0.0",
 | 
				
			||||||
 | 
					  "main": "index.js",
 | 
				
			||||||
 | 
					  "scripts": {
 | 
				
			||||||
 | 
					    "test": "echo \"Error: no test specified\" && exit 1",
 | 
				
			||||||
 | 
					    "bundle": "swagger-cli bundle -o ../src/bundled.yaml -t yaml ../src/main.yaml",
 | 
				
			||||||
 | 
					    "start": "npm run bundle; node main.js"
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
 | 
					  "keywords": [],
 | 
				
			||||||
 | 
					  "author": "",
 | 
				
			||||||
 | 
					  "license": "ISC",
 | 
				
			||||||
 | 
					  "description": "",
 | 
				
			||||||
 | 
					  "dependencies": {
 | 
				
			||||||
 | 
					    "express": "^4.21.2",
 | 
				
			||||||
 | 
					    "js-yaml": "^4.1.0",
 | 
				
			||||||
 | 
					    "package.json": "^2.0.1",
 | 
				
			||||||
 | 
					    "swagger-cli": "^4.0.4",
 | 
				
			||||||
 | 
					    "swagger-ui-express": "^5.0.1"
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										0
									
								
								front/Dockerfile
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										0
									
								
								front/Dockerfile
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							@@ -1,13 +1,13 @@
 | 
				
			|||||||
<!DOCTYPE html>
 | 
					<!doctype html>
 | 
				
			||||||
<html lang="">
 | 
					<html lang="">
 | 
				
			||||||
  <head>
 | 
					    <head>
 | 
				
			||||||
    <meta charset="UTF-8">
 | 
					        <meta charset="UTF-8" />
 | 
				
			||||||
    <link rel="icon" href="/favicon.ico">
 | 
					        <link rel="icon" href="/favicon.ico" />
 | 
				
			||||||
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
 | 
					        <meta name="viewport" content="width=device-width, initial-scale=1.0" />
 | 
				
			||||||
    <title>Vite App</title>
 | 
					        <title>Vite App</title>
 | 
				
			||||||
  </head>
 | 
					    </head>
 | 
				
			||||||
  <body>
 | 
					    <body>
 | 
				
			||||||
    <div id="app"></div>
 | 
					        <div id="app"></div>
 | 
				
			||||||
    <script type="module" src="/src/main.ts"></script>
 | 
					        <script type="module" src="/src/main.ts"></script>
 | 
				
			||||||
  </body>
 | 
					    </body>
 | 
				
			||||||
</html>
 | 
					</html>
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										10
									
								
								front/MyINPulse-front/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										10
									
								
								front/MyINPulse-front/package-lock.json
									
									
									
										generated
									
									
									
								
							@@ -10,6 +10,7 @@
 | 
				
			|||||||
      "dependencies": {
 | 
					      "dependencies": {
 | 
				
			||||||
        "axios": "^1.7.9",
 | 
					        "axios": "^1.7.9",
 | 
				
			||||||
        "cors": "^2.8.5",
 | 
					        "cors": "^2.8.5",
 | 
				
			||||||
 | 
					        "jwt-decode": "^4.0.0",
 | 
				
			||||||
        "keycloak-js": "^26.1.0",
 | 
					        "keycloak-js": "^26.1.0",
 | 
				
			||||||
        "pinia": "^2.3.1",
 | 
					        "pinia": "^2.3.1",
 | 
				
			||||||
        "pinia-plugin-persistedstate": "^4.2.0",
 | 
					        "pinia-plugin-persistedstate": "^4.2.0",
 | 
				
			||||||
@@ -3588,6 +3589,15 @@
 | 
				
			|||||||
        "graceful-fs": "^4.1.6"
 | 
					        "graceful-fs": "^4.1.6"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "node_modules/jwt-decode": {
 | 
				
			||||||
 | 
					      "version": "4.0.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-4.0.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-+KJGIyHgkGuIq3IEBNftfhW/LfWhXUIY6OmyVWjliu5KH1y0fw7VQ8YndE2O4qZdMSd9SqbnC8GOcZEy0Om7sA==",
 | 
				
			||||||
 | 
					      "license": "MIT",
 | 
				
			||||||
 | 
					      "engines": {
 | 
				
			||||||
 | 
					        "node": ">=18"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "node_modules/keycloak-js": {
 | 
					    "node_modules/keycloak-js": {
 | 
				
			||||||
      "version": "26.1.0",
 | 
					      "version": "26.1.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/keycloak-js/-/keycloak-js-26.1.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/keycloak-js/-/keycloak-js-26.1.0.tgz",
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -18,7 +18,8 @@
 | 
				
			|||||||
    "pinia": "^2.3.1",
 | 
					    "pinia": "^2.3.1",
 | 
				
			||||||
    "pinia-plugin-persistedstate": "^4.2.0",
 | 
					    "pinia-plugin-persistedstate": "^4.2.0",
 | 
				
			||||||
    "vue": "^3.5.13",
 | 
					    "vue": "^3.5.13",
 | 
				
			||||||
    "vue-router": "^4.5.0"
 | 
					    "vue-router": "^4.5.0",
 | 
				
			||||||
 | 
					    "jwt-decode": "^4.0.0"
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "devDependencies": {
 | 
					  "devDependencies": {
 | 
				
			||||||
    "@playwright/test": "^1.49.1",
 | 
					    "@playwright/test": "^1.49.1",
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										79
									
								
								front/MyINPulse-front/src/ApiClasses/Appointment.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								front/MyINPulse-front/src/ApiClasses/Appointment.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,79 @@
 | 
				
			|||||||
 | 
					// appointment.ts
 | 
				
			||||||
 | 
					class Appointment {
 | 
				
			||||||
 | 
					    private _idAppointment?: number;
 | 
				
			||||||
 | 
					    private _appointmentDate?: string;
 | 
				
			||||||
 | 
					    private _appointmentTime?: string;
 | 
				
			||||||
 | 
					    private _appointmentDuration?: string;
 | 
				
			||||||
 | 
					    private _appointmentPlace?: string;
 | 
				
			||||||
 | 
					    private _appointmentSubject?: string;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<Appointment> = {}) {
 | 
				
			||||||
 | 
					        this._idAppointment = data.idAppointment;
 | 
				
			||||||
 | 
					        this._appointmentDate = data.appointmentDate;
 | 
				
			||||||
 | 
					        this._appointmentTime = data.appointmentTime;
 | 
				
			||||||
 | 
					        this._appointmentDuration = data.appointmentDuration;
 | 
				
			||||||
 | 
					        this._appointmentPlace = data.appointmentPlace;
 | 
				
			||||||
 | 
					        this._appointmentSubject = data.appointmentSubject;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idAppointment(): number | undefined {
 | 
				
			||||||
 | 
					        return this._idAppointment;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set idAppointment(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._idAppointment = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get appointmentDate(): string | undefined {
 | 
				
			||||||
 | 
					        return this._appointmentDate;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set appointmentDate(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._appointmentDate = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get appointmentTime(): string | undefined {
 | 
				
			||||||
 | 
					        return this._appointmentTime;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set appointmentTime(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._appointmentTime = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get appointmentDuration(): string | undefined {
 | 
				
			||||||
 | 
					        return this._appointmentDuration;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set appointmentDuration(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._appointmentDuration = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get appointmentPlace(): string | undefined {
 | 
				
			||||||
 | 
					        return this._appointmentPlace;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set appointmentPlace(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._appointmentPlace = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get appointmentSubject(): string | undefined {
 | 
				
			||||||
 | 
					        return this._appointmentSubject;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set appointmentSubject(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._appointmentSubject = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            idAppointment: this.idAppointment,
 | 
				
			||||||
 | 
					            appointmentDate: this.appointmentDate,
 | 
				
			||||||
 | 
					            appointmentTime: this.appointmentTime,
 | 
				
			||||||
 | 
					            appointmentDuration: this.appointmentDuration,
 | 
				
			||||||
 | 
					            appointmentPlace: this.appointmentPlace,
 | 
				
			||||||
 | 
					            appointmentSubject: this.appointmentSubject,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default Appointment;
 | 
				
			||||||
							
								
								
									
										39
									
								
								front/MyINPulse-front/src/ApiClasses/JoinRequest.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								front/MyINPulse-front/src/ApiClasses/JoinRequest.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
				
			|||||||
 | 
					// joinRequest.ts
 | 
				
			||||||
 | 
					import UserEntrepreneur from "./UserEntrepreneur";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class JoinRequest {
 | 
				
			||||||
 | 
					    private _idProject?: number;
 | 
				
			||||||
 | 
					    private _entrepreneur?: UserEntrepreneur;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<JoinRequest> = {}) {
 | 
				
			||||||
 | 
					        this._idProject = data.idProject;
 | 
				
			||||||
 | 
					        this._entrepreneur = data.entrepreneur
 | 
				
			||||||
 | 
					            ? new UserEntrepreneur(data.entrepreneur)
 | 
				
			||||||
 | 
					            : undefined;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idProject(): number | undefined {
 | 
				
			||||||
 | 
					        return this._idProject;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set idProject(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._idProject = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get entrepreneur(): UserEntrepreneur | undefined {
 | 
				
			||||||
 | 
					        return this._entrepreneur;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set entrepreneur(value: UserEntrepreneur | undefined) {
 | 
				
			||||||
 | 
					        this._entrepreneur = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            idProject: this.idProject,
 | 
				
			||||||
 | 
					            entrepreneur: this.entrepreneur,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default JoinRequest;
 | 
				
			||||||
							
								
								
									
										24
									
								
								front/MyINPulse-front/src/ApiClasses/JoinRequestDecision.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								front/MyINPulse-front/src/ApiClasses/JoinRequestDecision.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
				
			|||||||
 | 
					// joinRequestDecision.ts
 | 
				
			||||||
 | 
					class JoinRequestDecision {
 | 
				
			||||||
 | 
					    private _isAccepted?: boolean;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<JoinRequestDecision> = {}) {
 | 
				
			||||||
 | 
					        this._isAccepted = data.isAccepted;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get isAccepted(): boolean | undefined {
 | 
				
			||||||
 | 
					        return this._isAccepted;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set isAccepted(value: boolean | undefined) {
 | 
				
			||||||
 | 
					        this._isAccepted = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            isAccepted: this._isAccepted,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default JoinRequestDecision;
 | 
				
			||||||
							
								
								
									
										89
									
								
								front/MyINPulse-front/src/ApiClasses/Project.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								front/MyINPulse-front/src/ApiClasses/Project.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,89 @@
 | 
				
			|||||||
 | 
					// project.ts
 | 
				
			||||||
 | 
					class Project {
 | 
				
			||||||
 | 
					    private _idProject?: number;
 | 
				
			||||||
 | 
					    private _projectName?: string;
 | 
				
			||||||
 | 
					    private _creationDate?: string;
 | 
				
			||||||
 | 
					    private _logo?: string;
 | 
				
			||||||
 | 
					    private _status?: "PENDING" | "ACTIVE" | "ENDED" | "ABORTED" | "REJECTED";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<Project> = {}) {
 | 
				
			||||||
 | 
					        this._idProject = data.idProject;
 | 
				
			||||||
 | 
					        this._projectName = data.projectName;
 | 
				
			||||||
 | 
					        this._creationDate = data.creationDate;
 | 
				
			||||||
 | 
					        this._logo = data.logo;
 | 
				
			||||||
 | 
					        this._status = data.status;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idProject(): number | undefined {
 | 
				
			||||||
 | 
					        return this._idProject;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set idProject(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._idProject = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get projectName(): string {
 | 
				
			||||||
 | 
					        return this._projectName ?? "";
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set projectName(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._projectName = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get creationDate(): string {
 | 
				
			||||||
 | 
					        return this._creationDate ?? "";
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set creationDate(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._creationDate = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get logo(): string | undefined {
 | 
				
			||||||
 | 
					        return this._logo;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set logo(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._logo = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get status():
 | 
				
			||||||
 | 
					        | "PENDING"
 | 
				
			||||||
 | 
					        | "ACTIVE"
 | 
				
			||||||
 | 
					        | "ENDED"
 | 
				
			||||||
 | 
					        | "ABORTED"
 | 
				
			||||||
 | 
					        | "REJECTED"
 | 
				
			||||||
 | 
					        | undefined {
 | 
				
			||||||
 | 
					        return this._status;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set status(
 | 
				
			||||||
 | 
					        value:
 | 
				
			||||||
 | 
					            | "PENDING"
 | 
				
			||||||
 | 
					            | "ACTIVE"
 | 
				
			||||||
 | 
					            | "ENDED"
 | 
				
			||||||
 | 
					            | "ABORTED"
 | 
				
			||||||
 | 
					            | "REJECTED"
 | 
				
			||||||
 | 
					            | undefined
 | 
				
			||||||
 | 
					    ) {
 | 
				
			||||||
 | 
					        this._status = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            idProject: this.idProject,
 | 
				
			||||||
 | 
					            projectName: this.projectName,
 | 
				
			||||||
 | 
					            creationDate: this.creationDate,
 | 
				
			||||||
 | 
					            logo: this.logo,
 | 
				
			||||||
 | 
					            status: this.status,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toCreatePayload() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            projectName: this.projectName,
 | 
				
			||||||
 | 
					            logo: this.logo,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default Project;
 | 
				
			||||||
							
								
								
									
										46
									
								
								front/MyINPulse-front/src/ApiClasses/ProjectDecision.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								front/MyINPulse-front/src/ApiClasses/ProjectDecision.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,46 @@
 | 
				
			|||||||
 | 
					// projectDecision.ts
 | 
				
			||||||
 | 
					class ProjectDecision {
 | 
				
			||||||
 | 
					    private _projectId?: number;
 | 
				
			||||||
 | 
					    private _adminId?: number;
 | 
				
			||||||
 | 
					    private _isAccepted?: boolean;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<ProjectDecision> = {}) {
 | 
				
			||||||
 | 
					        this._projectId = data.projectId;
 | 
				
			||||||
 | 
					        this._adminId = data.adminId;
 | 
				
			||||||
 | 
					        this._isAccepted = data.isAccepted;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get projectId(): number | undefined {
 | 
				
			||||||
 | 
					        return this._projectId;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set projectId(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._projectId = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get adminId(): number | undefined {
 | 
				
			||||||
 | 
					        return this._adminId;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set adminId(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._adminId = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get isAccepted(): boolean | undefined {
 | 
				
			||||||
 | 
					        return this._isAccepted;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set isAccepted(value: boolean | undefined) {
 | 
				
			||||||
 | 
					        this._isAccepted = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            projectId: this._projectId,
 | 
				
			||||||
 | 
					            adminId: this._adminId,
 | 
				
			||||||
 | 
					            isAccepted: this._isAccepted,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default ProjectDecision;
 | 
				
			||||||
							
								
								
									
										35
									
								
								front/MyINPulse-front/src/ApiClasses/Repport.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								front/MyINPulse-front/src/ApiClasses/Repport.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,35 @@
 | 
				
			|||||||
 | 
					// report.ts
 | 
				
			||||||
 | 
					class Report {
 | 
				
			||||||
 | 
					    private _idReport?: number;
 | 
				
			||||||
 | 
					    private _reportContent?: string;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<Report> = {}) {
 | 
				
			||||||
 | 
					        this._idReport = data.idReport;
 | 
				
			||||||
 | 
					        this._reportContent = data.reportContent;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idReport(): number | undefined {
 | 
				
			||||||
 | 
					        return this._idReport;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set idReport(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._idReport = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get reportContent(): string | undefined {
 | 
				
			||||||
 | 
					        return this._reportContent;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set reportContent(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._reportContent = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            idReport: this._idReport,
 | 
				
			||||||
 | 
					            reportContent: this._reportContent,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default Report;
 | 
				
			||||||
							
								
								
									
										57
									
								
								front/MyINPulse-front/src/ApiClasses/SectionCell.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								front/MyINPulse-front/src/ApiClasses/SectionCell.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,57 @@
 | 
				
			|||||||
 | 
					// sectionCell.ts
 | 
				
			||||||
 | 
					class SectionCell {
 | 
				
			||||||
 | 
					    private _idSectionCell?: number;
 | 
				
			||||||
 | 
					    private _sectionId?: number;
 | 
				
			||||||
 | 
					    private _contentSectionCell?: string;
 | 
				
			||||||
 | 
					    private _modificationDate?: string;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<SectionCell> = {}) {
 | 
				
			||||||
 | 
					        this._idSectionCell = data.idSectionCell;
 | 
				
			||||||
 | 
					        this._sectionId = data.sectionId;
 | 
				
			||||||
 | 
					        this._contentSectionCell = data.contentSectionCell;
 | 
				
			||||||
 | 
					        this._modificationDate = data.modificationDate;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idSectionCell(): number | undefined {
 | 
				
			||||||
 | 
					        return this._idSectionCell;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set idSectionCell(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._idSectionCell = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get sectionId(): number | undefined {
 | 
				
			||||||
 | 
					        return this._sectionId;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set sectionId(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._sectionId = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get contentSectionCell(): string | undefined {
 | 
				
			||||||
 | 
					        return this._contentSectionCell;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set contentSectionCell(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._contentSectionCell = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get modificationDate(): string | undefined {
 | 
				
			||||||
 | 
					        return this._modificationDate;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set modificationDate(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._modificationDate = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            idSectionCell: this._idSectionCell,
 | 
				
			||||||
 | 
					            sectionId: this._sectionId,
 | 
				
			||||||
 | 
					            contentSectionCell: this._contentSectionCell,
 | 
				
			||||||
 | 
					            modificationDate: this._modificationDate,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default SectionCell;
 | 
				
			||||||
							
								
								
									
										78
									
								
								front/MyINPulse-front/src/ApiClasses/User.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								front/MyINPulse-front/src/ApiClasses/User.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,78 @@
 | 
				
			|||||||
 | 
					class User {
 | 
				
			||||||
 | 
					    private _idUser?: number;
 | 
				
			||||||
 | 
					    private _userSurname?: string;
 | 
				
			||||||
 | 
					    private _userName?: string;
 | 
				
			||||||
 | 
					    private _primaryMail?: string;
 | 
				
			||||||
 | 
					    private _secondaryMail?: string;
 | 
				
			||||||
 | 
					    private _phoneNumber?: string;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<User> = {}) {
 | 
				
			||||||
 | 
					        this._idUser = data.idUser;
 | 
				
			||||||
 | 
					        this._userSurname = data.userSurname;
 | 
				
			||||||
 | 
					        this._userName = data.userName;
 | 
				
			||||||
 | 
					        this._primaryMail = data.primaryMail;
 | 
				
			||||||
 | 
					        this._secondaryMail = data.secondaryMail;
 | 
				
			||||||
 | 
					        this._phoneNumber = data.phoneNumber;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idUser(): number | undefined {
 | 
				
			||||||
 | 
					        return this._idUser;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set idUser(value: number | undefined) {
 | 
				
			||||||
 | 
					        this._idUser = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get userSurname(): string | undefined {
 | 
				
			||||||
 | 
					        return this._userSurname;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set userSurname(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._userSurname = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get userName(): string | undefined {
 | 
				
			||||||
 | 
					        return this._userName;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set userName(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._userName = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get primaryMail(): string | undefined {
 | 
				
			||||||
 | 
					        return this._primaryMail;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set primaryMail(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._primaryMail = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get secondaryMail(): string | undefined {
 | 
				
			||||||
 | 
					        return this._secondaryMail;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set secondaryMail(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._secondaryMail = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get phoneNumber(): string | undefined {
 | 
				
			||||||
 | 
					        return this._phoneNumber;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set phoneNumber(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._phoneNumber = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            idUser: this._idUser,
 | 
				
			||||||
 | 
					            userSurname: this._userSurname,
 | 
				
			||||||
 | 
					            userName: this._userName,
 | 
				
			||||||
 | 
					            primaryMail: this._primaryMail,
 | 
				
			||||||
 | 
					            secondaryMail: this._secondaryMail,
 | 
				
			||||||
 | 
					            phoneNumber: this._phoneNumber,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default User;
 | 
				
			||||||
							
								
								
									
										14
									
								
								front/MyINPulse-front/src/ApiClasses/UserAdmin.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								front/MyINPulse-front/src/ApiClasses/UserAdmin.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
				
			|||||||
 | 
					// user-admin.ts
 | 
				
			||||||
 | 
					import User from "./User";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class UserAdmin extends User {
 | 
				
			||||||
 | 
					    constructor(data: Partial<UserAdmin> = {}) {
 | 
				
			||||||
 | 
					        super(data);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get idUser(): number | undefined {
 | 
				
			||||||
 | 
					        return super.idUser;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default UserAdmin;
 | 
				
			||||||
							
								
								
									
										50
									
								
								front/MyINPulse-front/src/ApiClasses/UserEntrepreneur.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								front/MyINPulse-front/src/ApiClasses/UserEntrepreneur.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,50 @@
 | 
				
			|||||||
 | 
					// user-entrepreneur.ts
 | 
				
			||||||
 | 
					import User from "./User";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class UserEntrepreneur extends User {
 | 
				
			||||||
 | 
					    private _school?: string;
 | 
				
			||||||
 | 
					    private _course?: string;
 | 
				
			||||||
 | 
					    private _sneeStatus?: boolean;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    constructor(data: Partial<UserEntrepreneur> = {}) {
 | 
				
			||||||
 | 
					        super(data);
 | 
				
			||||||
 | 
					        this._school = data.school;
 | 
				
			||||||
 | 
					        this._course = data.course;
 | 
				
			||||||
 | 
					        this._sneeStatus = data.sneeStatus;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get school(): string | undefined {
 | 
				
			||||||
 | 
					        return this._school;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set school(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._school = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get course(): string | undefined {
 | 
				
			||||||
 | 
					        return this._course;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set course(value: string | undefined) {
 | 
				
			||||||
 | 
					        this._course = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    get sneeStatus(): boolean | undefined {
 | 
				
			||||||
 | 
					        return this._sneeStatus;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    set sneeStatus(value: boolean | undefined) {
 | 
				
			||||||
 | 
					        this._sneeStatus = value;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    toObject() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            ...super.toObject(),
 | 
				
			||||||
 | 
					            school: this._school,
 | 
				
			||||||
 | 
					            course: this._course,
 | 
				
			||||||
 | 
					            sneeStatus: this._sneeStatus,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default UserEntrepreneur;
 | 
				
			||||||
@@ -1,47 +1,12 @@
 | 
				
			|||||||
<script setup lang="ts">
 | 
					<script setup lang="ts">
 | 
				
			||||||
import { RouterView } from "vue-router";
 | 
					import { /*RouterLink,*/ RouterView } from "vue-router";
 | 
				
			||||||
import ErrorWrapper from "@/views/errorWrapper.vue";
 | 
					import ErrorWrapper from "@/views/errorWrapper.vue";
 | 
				
			||||||
import ProjectComponent from "@/components/ProjectComponent.vue";
 | 
					 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<template>
 | 
					<template>
 | 
				
			||||||
    <HeaderComponent />
 | 
					    <Header />
 | 
				
			||||||
    <error-wrapper></error-wrapper>
 | 
					    <ErrorWrapper />
 | 
				
			||||||
    <div id="main">
 | 
					    <!--<RouterLink to="/">Home</RouterLink> | -->
 | 
				
			||||||
        <ProjectComponent
 | 
					    <!--<RouterLink to="/canvas">Canvas</RouterLink> -->
 | 
				
			||||||
            v-for="(project, index) in projects"
 | 
					 | 
				
			||||||
            :key="index"
 | 
					 | 
				
			||||||
            :project-name="project.name"
 | 
					 | 
				
			||||||
        />
 | 
					 | 
				
			||||||
    </div>
 | 
					 | 
				
			||||||
    <RouterView />
 | 
					    <RouterView />
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					 | 
				
			||||||
<script lang="ts">
 | 
					 | 
				
			||||||
import HeaderComponent from "@/components/HeaderComponent.vue";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default {
 | 
					 | 
				
			||||||
    name: "App",
 | 
					 | 
				
			||||||
    components: {
 | 
					 | 
				
			||||||
        HeaderComponent,
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    data() {
 | 
					 | 
				
			||||||
        return {
 | 
					 | 
				
			||||||
            projects: [
 | 
					 | 
				
			||||||
                {
 | 
					 | 
				
			||||||
                    name: "Projet Alpha",
 | 
					 | 
				
			||||||
                    //link: './project-alpha.html',
 | 
					 | 
				
			||||||
                    //members: ['Alice', 'Bob', 'Charlie'],
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
                {
 | 
					 | 
				
			||||||
                    name: "Projet Beta",
 | 
					 | 
				
			||||||
                    //link: './project-beta.html',
 | 
					 | 
				
			||||||
                    //members: ['David', 'Eve', 'Frank'],
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
        };
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
</script>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<style scoped></style>
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										98
									
								
								front/MyINPulse-front/src/components/AddProjectForm.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								front/MyINPulse-front/src/components/AddProjectForm.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,98 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <form class="add-project-form" @submit.prevent="submitProject">
 | 
				
			||||||
 | 
					        <h2>Ajouter un projet</h2>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <div class="form-group">
 | 
				
			||||||
 | 
					            <label for="projectName">Nom du projet</label>
 | 
				
			||||||
 | 
					            <input
 | 
				
			||||||
 | 
					                id="projectName"
 | 
				
			||||||
 | 
					                v-model="project.projectName"
 | 
				
			||||||
 | 
					                type="text"
 | 
				
			||||||
 | 
					                required
 | 
				
			||||||
 | 
					            />
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <div class="form-group">
 | 
				
			||||||
 | 
					            <label for="logo">Logo</label>
 | 
				
			||||||
 | 
					            <input
 | 
				
			||||||
 | 
					                id="logo"
 | 
				
			||||||
 | 
					                v-model="project.logo"
 | 
				
			||||||
 | 
					                type="text"
 | 
				
			||||||
 | 
					                placeholder="(Description)"
 | 
				
			||||||
 | 
					            />
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <button type="submit">Ajouter</button>
 | 
				
			||||||
 | 
					    </form>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { ref } from "vue";
 | 
				
			||||||
 | 
					import Project from "@/ApiClasses/Project";
 | 
				
			||||||
 | 
					import { addProjectManually } from "@/services/Apis/Admin";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const project = ref(new Project({}));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function submitProject() {
 | 
				
			||||||
 | 
					    addProjectManually(
 | 
				
			||||||
 | 
					        project.value.toCreatePayload(),
 | 
				
			||||||
 | 
					        (res) => console.log("Success:", res.data),
 | 
				
			||||||
 | 
					        (err) => console.error("Error:", err)
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					h2 {
 | 
				
			||||||
 | 
					    font-size: 1.5rem;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					    margin-bottom: 1.2rem;
 | 
				
			||||||
 | 
					    border-bottom: 2px solid #ddd;
 | 
				
			||||||
 | 
					    padding-bottom: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.add-project-form {
 | 
				
			||||||
 | 
					    max-width: 500px;
 | 
				
			||||||
 | 
					    margin: 0 auto;
 | 
				
			||||||
 | 
					    padding: 20px;
 | 
				
			||||||
 | 
					    background: #fff;
 | 
				
			||||||
 | 
					    border-radius: 10px;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					h2 {
 | 
				
			||||||
 | 
					    margin-bottom: 20px;
 | 
				
			||||||
 | 
					    font-size: 24px;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.form-group {
 | 
				
			||||||
 | 
					    margin-bottom: 15px;
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-direction: column;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					label {
 | 
				
			||||||
 | 
					    font-weight: bold;
 | 
				
			||||||
 | 
					    margin-bottom: 5px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					input {
 | 
				
			||||||
 | 
					    padding: 8px;
 | 
				
			||||||
 | 
					    border-radius: 5px;
 | 
				
			||||||
 | 
					    border: 1px solid #ccc;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button {
 | 
				
			||||||
 | 
					    background-color: #4caf50;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    padding: 10px 15px;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 5px;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button:hover {
 | 
				
			||||||
 | 
					    background-color: #45a049;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										192
									
								
								front/MyINPulse-front/src/components/AdminAppointments.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								front/MyINPulse-front/src/components/AdminAppointments.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,192 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div>
 | 
				
			||||||
 | 
					        <h2>Ajouter / Modifier un Rapport pour un Rendez-vous</h2>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <form @submit.prevent="submitReport">
 | 
				
			||||||
 | 
					            <div>
 | 
				
			||||||
 | 
					                <label for="appointmentId">ID du rendez-vous :</label>
 | 
				
			||||||
 | 
					                <div style="display: flex; align-items: center; gap: 0.5rem">
 | 
				
			||||||
 | 
					                    <button type="button" @click="decrementId">-</button>
 | 
				
			||||||
 | 
					                    <input
 | 
				
			||||||
 | 
					                        id="appointmentId"
 | 
				
			||||||
 | 
					                        v-model.number="appointmentId"
 | 
				
			||||||
 | 
					                        type="number"
 | 
				
			||||||
 | 
					                        min="1"
 | 
				
			||||||
 | 
					                        style="width: 60px; text-align: center"
 | 
				
			||||||
 | 
					                    />
 | 
				
			||||||
 | 
					                    <button type="button" @click="incrementId">+</button>
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div>
 | 
				
			||||||
 | 
					                <label for="reportContent">Contenu du rapport :</label>
 | 
				
			||||||
 | 
					                <textarea
 | 
				
			||||||
 | 
					                    id="reportContent"
 | 
				
			||||||
 | 
					                    v-model="reportContent"
 | 
				
			||||||
 | 
					                    required
 | 
				
			||||||
 | 
					                ></textarea>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <button type="submit">
 | 
				
			||||||
 | 
					                {{ isUpdate ? "Mettre à jour" : "Créer" }} le rapport
 | 
				
			||||||
 | 
					            </button>
 | 
				
			||||||
 | 
					        </form>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <p v-if="responseMessage">{{ responseMessage }}</p>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script>
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					    createAppointmentReport,
 | 
				
			||||||
 | 
					    updateAppointmentReport,
 | 
				
			||||||
 | 
					} from "@/services/Apis/Admin";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default {
 | 
				
			||||||
 | 
					    name: "AdminAppointmentsComponent",
 | 
				
			||||||
 | 
					    data() {
 | 
				
			||||||
 | 
					        return {
 | 
				
			||||||
 | 
					            appointmentId: 1,
 | 
				
			||||||
 | 
					            reportContent: "",
 | 
				
			||||||
 | 
					            responseMessage: "",
 | 
				
			||||||
 | 
					            isUpdate: false,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    watch: {
 | 
				
			||||||
 | 
					        appointmentId(newVal) {
 | 
				
			||||||
 | 
					            if (newVal) {
 | 
				
			||||||
 | 
					                this.isUpdate = true;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    methods: {
 | 
				
			||||||
 | 
					        incrementId() {
 | 
				
			||||||
 | 
					            this.appointmentId++;
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        decrementId() {
 | 
				
			||||||
 | 
					            if (this.appointmentId > 1) {
 | 
				
			||||||
 | 
					                this.appointmentId--;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        submitReport() {
 | 
				
			||||||
 | 
					            const reportData = { content: this.reportContent };
 | 
				
			||||||
 | 
					            const onSuccess = (response) => {
 | 
				
			||||||
 | 
					                if (response.status === 201 || response.status === 200) {
 | 
				
			||||||
 | 
					                    this.responseMessage =
 | 
				
			||||||
 | 
					                        "Rapport " +
 | 
				
			||||||
 | 
					                        (this.isUpdate ? "mis à jour" : "créé") +
 | 
				
			||||||
 | 
					                        " avec succès.";
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            };
 | 
				
			||||||
 | 
					            const onError = (error) => {
 | 
				
			||||||
 | 
					                console.error(
 | 
				
			||||||
 | 
					                    "Erreur lors de l'envoi du rapport :",
 | 
				
			||||||
 | 
					                    error.response || error.message
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					                if (error.response && error.response.status === 400) {
 | 
				
			||||||
 | 
					                    this.responseMessage =
 | 
				
			||||||
 | 
					                        "Requête invalide. Vérifiez les informations.";
 | 
				
			||||||
 | 
					                } else if (error.response && error.response.status === 401) {
 | 
				
			||||||
 | 
					                    this.responseMessage =
 | 
				
			||||||
 | 
					                        "Vous n'êtes pas autorisé à effectuer cette action.";
 | 
				
			||||||
 | 
					                } else {
 | 
				
			||||||
 | 
					                    this.responseMessage =
 | 
				
			||||||
 | 
					                        "Une erreur est survenue. Veuillez réessayer.";
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (this.isUpdate) {
 | 
				
			||||||
 | 
					                updateAppointmentReport(
 | 
				
			||||||
 | 
					                    this.appointmentId,
 | 
				
			||||||
 | 
					                    reportData,
 | 
				
			||||||
 | 
					                    onSuccess,
 | 
				
			||||||
 | 
					                    onError
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                createAppointmentReport(
 | 
				
			||||||
 | 
					                    this.appointmentId,
 | 
				
			||||||
 | 
					                    reportData,
 | 
				
			||||||
 | 
					                    onSuccess,
 | 
				
			||||||
 | 
					                    onError
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					/* Centrer le formulaire */
 | 
				
			||||||
 | 
					div {
 | 
				
			||||||
 | 
					    max-width: 600px;
 | 
				
			||||||
 | 
					    margin: 0 auto;
 | 
				
			||||||
 | 
					    padding: 1rem;
 | 
				
			||||||
 | 
					    background-color: #f9f9f9;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Titre */
 | 
				
			||||||
 | 
					h2 {
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Formulaire */
 | 
				
			||||||
 | 
					form {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-direction: column;
 | 
				
			||||||
 | 
					    gap: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Boutons */
 | 
				
			||||||
 | 
					button {
 | 
				
			||||||
 | 
					    padding: 0.5rem 1rem;
 | 
				
			||||||
 | 
					    background-color: #007bff;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 4px;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    transition: background-color 0.3s ease;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button:hover {
 | 
				
			||||||
 | 
					    background-color: #0056b3;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Input et Textarea */
 | 
				
			||||||
 | 
					input[type="number"],
 | 
				
			||||||
 | 
					textarea {
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    padding: 0.5rem;
 | 
				
			||||||
 | 
					    border: 1px solid #ccc;
 | 
				
			||||||
 | 
					    border-radius: 4px;
 | 
				
			||||||
 | 
					    font-size: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					textarea {
 | 
				
			||||||
 | 
					    height: 120px;
 | 
				
			||||||
 | 
					    resize: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Message de réponse */
 | 
				
			||||||
 | 
					p {
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    font-weight: bold;
 | 
				
			||||||
 | 
					    color: green;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Boutons d'incrémentation/décrémentation */
 | 
				
			||||||
 | 
					div[style*="display: flex"] button {
 | 
				
			||||||
 | 
					    background-color: #6c757d;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 4px;
 | 
				
			||||||
 | 
					    padding: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					div[style*="display: flex"] button:hover {
 | 
				
			||||||
 | 
					    background-color: #5a6268;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										93
									
								
								front/MyINPulse-front/src/components/AgendaComponent.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								front/MyINPulse-front/src/components/AgendaComponent.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,93 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div id="agenda">
 | 
				
			||||||
 | 
					        <h3>Rendez-vous</h3>
 | 
				
			||||||
 | 
					        <table>
 | 
				
			||||||
 | 
					            <thead>
 | 
				
			||||||
 | 
					                <tr>
 | 
				
			||||||
 | 
					                    <th>Projet</th>
 | 
				
			||||||
 | 
					                    <th>Date</th>
 | 
				
			||||||
 | 
					                    <th>Lieu</th>
 | 
				
			||||||
 | 
					                </tr>
 | 
				
			||||||
 | 
					            </thead>
 | 
				
			||||||
 | 
					            <tbody>
 | 
				
			||||||
 | 
					                <tr v-for="(p, index) in projectRDV" :key="index">
 | 
				
			||||||
 | 
					                    <td>{{ p.projectName }}</td>
 | 
				
			||||||
 | 
					                    <td>{{ p.date }}</td>
 | 
				
			||||||
 | 
					                    <td>{{ p.lieu }}</td>
 | 
				
			||||||
 | 
					                </tr>
 | 
				
			||||||
 | 
					            </tbody>
 | 
				
			||||||
 | 
					        </table>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { defineProps } from "vue";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					interface rendezVous {
 | 
				
			||||||
 | 
					    projectName: string;
 | 
				
			||||||
 | 
					    date: string;
 | 
				
			||||||
 | 
					    lieu: string;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					defineProps<{
 | 
				
			||||||
 | 
					    projectRDV: rendezVous[];
 | 
				
			||||||
 | 
					}>();
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					h3 {
 | 
				
			||||||
 | 
					    font-size: 1.5rem;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					    margin-bottom: 1.2rem;
 | 
				
			||||||
 | 
					    border-bottom: 2px solid #ddd;
 | 
				
			||||||
 | 
					    padding-bottom: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#agenda {
 | 
				
			||||||
 | 
					    padding: 20px;
 | 
				
			||||||
 | 
					    background-color: white;
 | 
				
			||||||
 | 
					    border-radius: 10px;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 6px rgba(0, 0, 0, 0.05);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Table Styling */
 | 
				
			||||||
 | 
					table {
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    border-collapse: collapse;
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					    text-align: left;
 | 
				
			||||||
 | 
					    margin-top: 20px;
 | 
				
			||||||
 | 
					    border: 1px solid #ccc;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					th {
 | 
				
			||||||
 | 
					    background-color: #f0f2f5;
 | 
				
			||||||
 | 
					    padding: 12px;
 | 
				
			||||||
 | 
					    font-weight: 600;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Table Body Rows */
 | 
				
			||||||
 | 
					tbody tr {
 | 
				
			||||||
 | 
					    border-bottom: 1px solid #ddd;
 | 
				
			||||||
 | 
					    transition: background-color 0.2s ease; /* Smooth hover effect */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					tbody tr:hover {
 | 
				
			||||||
 | 
					    background-color: #f9f9f9; /* Highlight row on hover */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Cells Styling */
 | 
				
			||||||
 | 
					td {
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					    border: 1px solid #eee;
 | 
				
			||||||
 | 
					    font-size: 14px;
 | 
				
			||||||
 | 
					    vertical-align: middle; /* Align text to middle */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* First Column Styling */
 | 
				
			||||||
 | 
					td:first-child {
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    width: 50px; /* Adjust width as needed */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										77
									
								
								front/MyINPulse-front/src/components/AllEntrep.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								front/MyINPulse-front/src/components/AllEntrep.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,77 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div class="entrepreneur-list">
 | 
				
			||||||
 | 
					        <button @click="fetchEntrepreneurs">Afficher les entrepreneurs</button>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <ul v-if="entrepreneurs.length">
 | 
				
			||||||
 | 
					            <li
 | 
				
			||||||
 | 
					                v-for="entrepreneur in entrepreneurs"
 | 
				
			||||||
 | 
					                :key="entrepreneur.idUser"
 | 
				
			||||||
 | 
					            >
 | 
				
			||||||
 | 
					                {{ entrepreneur.userName }} {{ entrepreneur.userSurname }} -
 | 
				
			||||||
 | 
					                {{ entrepreneur.primaryMail }}
 | 
				
			||||||
 | 
					            </li>
 | 
				
			||||||
 | 
					        </ul>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <p v-else-if="loading">Chargement...</p>
 | 
				
			||||||
 | 
					        <p v-else-if="errorMessage">{{ errorMessage }}</p>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { ref } from "vue";
 | 
				
			||||||
 | 
					import type { AxiosResponse, AxiosError } from "axios";
 | 
				
			||||||
 | 
					import { getAllEntrepreneurs } from "@/services/Apis/Unauth"; // ajuste ce chemin selon ton projet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Types
 | 
				
			||||||
 | 
					interface Entrepreneur {
 | 
				
			||||||
 | 
					    idUser: number;
 | 
				
			||||||
 | 
					    userName: string;
 | 
				
			||||||
 | 
					    userSurname: string;
 | 
				
			||||||
 | 
					    primaryMail: string;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Refs
 | 
				
			||||||
 | 
					const entrepreneurs = ref<Entrepreneur[]>([]);
 | 
				
			||||||
 | 
					const loading = ref(false);
 | 
				
			||||||
 | 
					const errorMessage = ref("");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Méthode
 | 
				
			||||||
 | 
					function fetchEntrepreneurs() {
 | 
				
			||||||
 | 
					    loading.value = true;
 | 
				
			||||||
 | 
					    errorMessage.value = "";
 | 
				
			||||||
 | 
					    getAllEntrepreneurs(
 | 
				
			||||||
 | 
					        (response: AxiosResponse) => {
 | 
				
			||||||
 | 
					            if (Array.isArray(response.data)) {
 | 
				
			||||||
 | 
					                entrepreneurs.value = response.data;
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                console.error("Format inattendu :", response.data);
 | 
				
			||||||
 | 
					                errorMessage.value = "Réponse inattendue du serveur.";
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            loading.value = false;
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        (error: AxiosError) => {
 | 
				
			||||||
 | 
					            errorMessage.value = "Erreur lors du chargement des entrepreneurs.";
 | 
				
			||||||
 | 
					            console.error(error);
 | 
				
			||||||
 | 
					            loading.value = false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					button {
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					    padding: 0.5rem 1rem;
 | 
				
			||||||
 | 
					    font-size: 1rem;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ul {
 | 
				
			||||||
 | 
					    list-style-type: none;
 | 
				
			||||||
 | 
					    padding: 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					li {
 | 
				
			||||||
 | 
					    margin-bottom: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										90
									
								
								front/MyINPulse-front/src/components/GetAppointments.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								front/MyINPulse-front/src/components/GetAppointments.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,90 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div class="appointment-list">
 | 
				
			||||||
 | 
					        <h2>Liste des rendez-vous</h2>
 | 
				
			||||||
 | 
					        <h3>de {{ store.user.username }}</h3>
 | 
				
			||||||
 | 
					        <ul v-if="appointments.length">
 | 
				
			||||||
 | 
					            <li v-for="appt in appointments" :key="appt.idAppointment">
 | 
				
			||||||
 | 
					                <strong>Sujet :</strong> {{ appt.appointmentSubject }}<br />
 | 
				
			||||||
 | 
					                <strong>Date :</strong> {{ appt.appointmentDate }}<br />
 | 
				
			||||||
 | 
					                <strong>Heure :</strong> {{ appt.appointmentTime }}<br />
 | 
				
			||||||
 | 
					                <strong>Durée :</strong> {{ appt.appointmentDuration }}<br />
 | 
				
			||||||
 | 
					                <strong>Lieu :</strong> {{ appt.appointmentPlace }}
 | 
				
			||||||
 | 
					                <hr />
 | 
				
			||||||
 | 
					            </li>
 | 
				
			||||||
 | 
					        </ul>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <p v-else>Aucun rendez-vous trouvé.</p>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { onMounted, ref } from "vue";
 | 
				
			||||||
 | 
					import { callApi } from "@/services/api";
 | 
				
			||||||
 | 
					import Appointment from "@/ApiClasses/Appointment";
 | 
				
			||||||
 | 
					import { store } from "@/main.ts";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Define the interface for the API response
 | 
				
			||||||
 | 
					interface AppointmentResponse {
 | 
				
			||||||
 | 
					    idAppointment: number;
 | 
				
			||||||
 | 
					    appointmentSubject: string;
 | 
				
			||||||
 | 
					    appointmentDate: string;
 | 
				
			||||||
 | 
					    appointmentTime: string;
 | 
				
			||||||
 | 
					    appointmentDuration: string;
 | 
				
			||||||
 | 
					    appointmentPlace: string;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const appointments = ref<Appointment[]>([]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function loadAppointments() {
 | 
				
			||||||
 | 
					    if (!store.user) {
 | 
				
			||||||
 | 
					        console.error(
 | 
				
			||||||
 | 
					            "L'utilisateur n'est pas connecté ou les données utilisateur ne sont pas disponibles."
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    //console.log("username :", store.user.username);
 | 
				
			||||||
 | 
					    //console.log("token  :", store.user.token);
 | 
				
			||||||
 | 
					    callApi(
 | 
				
			||||||
 | 
					        "/admin/appointments/upcoming",
 | 
				
			||||||
 | 
					        (response) => {
 | 
				
			||||||
 | 
					            appointments.value = response.data.map(
 | 
				
			||||||
 | 
					                (item: AppointmentResponse) => new Appointment(item)
 | 
				
			||||||
 | 
					            );
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        (error) => {
 | 
				
			||||||
 | 
					            console.error(
 | 
				
			||||||
 | 
					                "Erreur lors de la récupération des rendez-vous :",
 | 
				
			||||||
 | 
					                error
 | 
				
			||||||
 | 
					            );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onMounted(() => {
 | 
				
			||||||
 | 
					    loadAppointments();
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					.appointment-list {
 | 
				
			||||||
 | 
					    max-width: 600px;
 | 
				
			||||||
 | 
					    margin: 0 auto;
 | 
				
			||||||
 | 
					    padding: 20px;
 | 
				
			||||||
 | 
					    background: #fdfdfd;
 | 
				
			||||||
 | 
					    border-radius: 10px;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 10px rgba(0, 0, 0, 0.08);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					h2 {
 | 
				
			||||||
 | 
					    font-size: 1.8rem;
 | 
				
			||||||
 | 
					    margin-bottom: 20px;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					    border-bottom: 2px solid #ddd;
 | 
				
			||||||
 | 
					    padding-bottom: 10px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					li {
 | 
				
			||||||
 | 
					    margin-bottom: 15px;
 | 
				
			||||||
 | 
					    line-height: 1.6;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
@@ -1,26 +1,64 @@
 | 
				
			|||||||
<template>
 | 
					<template>
 | 
				
			||||||
    <header>
 | 
					    <header class="header">
 | 
				
			||||||
        <img src="./icons/logo inpulse.png" alt="INPulse" />
 | 
					        <a
 | 
				
			||||||
 | 
					            href="https://www.bordeaux-inp.fr/fr/lincubateur-bordeaux-inpulse"
 | 
				
			||||||
 | 
					            target="_blank"
 | 
				
			||||||
 | 
					            rel="noopener"
 | 
				
			||||||
 | 
					        >
 | 
				
			||||||
 | 
					            <img
 | 
				
			||||||
 | 
					                src="./icons/logo inpulse.png"
 | 
				
			||||||
 | 
					                alt="INPulse Logo"
 | 
				
			||||||
 | 
					                class="logo"
 | 
				
			||||||
 | 
					            />
 | 
				
			||||||
 | 
					        </a>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <div class="header-actions">
 | 
				
			||||||
 | 
					            <button class="logout" @click="store.logout">Logout</button>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
    </header>
 | 
					    </header>
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<script lang="ts">
 | 
					<script setup lang="ts">
 | 
				
			||||||
export default {
 | 
					import { store } from "@/main.ts";
 | 
				
			||||||
    name: "HeaderComponent",
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<style scoped>
 | 
					<style scoped>
 | 
				
			||||||
header img {
 | 
					@import "@/components/canvas/style-project.css";
 | 
				
			||||||
    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: 15px 30px;
 | 
				
			||||||
    background-color: #fff;
 | 
					    background-color: #f9f9f9;
 | 
				
			||||||
    border-bottom: 2px solid #ddd;
 | 
					    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.logo {
 | 
				
			||||||
 | 
					    height: 50px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.header-actions {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    gap: 20px;
 | 
				
			||||||
 | 
					    position: relative;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.logout {
 | 
				
			||||||
 | 
					    background-color: #009cde;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    padding: 10px 15px;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    font-size: 14px;
 | 
				
			||||||
 | 
					    border-radius: 5px;
 | 
				
			||||||
 | 
					    text-decoration: none;
 | 
				
			||||||
 | 
					    transition: background-color 0.2s ease;
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.logout:hover {
 | 
				
			||||||
 | 
					    background-color: #007bad;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
</style>
 | 
					</style>
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										229
									
								
								front/MyINPulse-front/src/components/LoginComponent.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										229
									
								
								front/MyINPulse-front/src/components/LoginComponent.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,229 @@
 | 
				
			|||||||
 | 
					<script lang="ts" setup>
 | 
				
			||||||
 | 
					import { onMounted } from "vue";
 | 
				
			||||||
 | 
					import { useRouter } from "vue-router";
 | 
				
			||||||
 | 
					import { jwtDecode } from "jwt-decode"; // i hope this doesn't break the code later
 | 
				
			||||||
 | 
					import { store } from "../main.ts";
 | 
				
			||||||
 | 
					import { checkPending } from "@/services/Apis/Unauth";
 | 
				
			||||||
 | 
					import Header from "@/components/HeaderComponent.vue";
 | 
				
			||||||
 | 
					const router = useRouter();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type TokenPayload = {
 | 
				
			||||||
 | 
					    realm_access?: {
 | 
				
			||||||
 | 
					        roles?: string[];
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//const customRequest = ref("");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onMounted(() => {
 | 
				
			||||||
 | 
					    if (store.authenticated && store.user.token) {
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            const decoded = jwtDecode<TokenPayload>(store.user.token);
 | 
				
			||||||
 | 
					            const roles = decoded.realm_access?.roles || [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (roles.includes("MyINPulse-admin")) {
 | 
				
			||||||
 | 
					                router.push("/admin");
 | 
				
			||||||
 | 
					                return;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (roles.includes("MyINPulse-entrepreneur")) {
 | 
				
			||||||
 | 
					                router.push("/canvas");
 | 
				
			||||||
 | 
					                return;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            checkPending(
 | 
				
			||||||
 | 
					                (response) => {
 | 
				
			||||||
 | 
					                    const isValidated = response.data === true;
 | 
				
			||||||
 | 
					                    if (
 | 
				
			||||||
 | 
					                        isValidated &&
 | 
				
			||||||
 | 
					                        roles.includes("MyINPulse-entrepreneur")
 | 
				
			||||||
 | 
					                    ) {
 | 
				
			||||||
 | 
					                        router.push("/canvas");
 | 
				
			||||||
 | 
					                        //router.push("/JorCproject");
 | 
				
			||||||
 | 
					                    } else {
 | 
				
			||||||
 | 
					                        router.push("/JorCproject");
 | 
				
			||||||
 | 
					                        //router.push("/finalize");
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                },
 | 
				
			||||||
 | 
					                (error) => {
 | 
				
			||||||
 | 
					                    if (error.response?.status === 403) {
 | 
				
			||||||
 | 
					                        router.push("/finalize");
 | 
				
			||||||
 | 
					                    } else {
 | 
				
			||||||
 | 
					                        console.error(
 | 
				
			||||||
 | 
					                            "Unexpected error during checkPending",
 | 
				
			||||||
 | 
					                            error
 | 
				
			||||||
 | 
					                        );
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            );
 | 
				
			||||||
 | 
					        } catch (err) {
 | 
				
			||||||
 | 
					            console.error("Failed to decode token", err);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <Header />
 | 
				
			||||||
 | 
					    <error-wrapper></error-wrapper>
 | 
				
			||||||
 | 
					    <div class="auth-container">
 | 
				
			||||||
 | 
					        <div class="auth-card">
 | 
				
			||||||
 | 
					            <h1>Bienvenue à MyINPulse</h1>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div
 | 
				
			||||||
 | 
					                class="status"
 | 
				
			||||||
 | 
					                :class="store.authenticated ? 'success' : 'error'"
 | 
				
			||||||
 | 
					            >
 | 
				
			||||||
 | 
					                <p>
 | 
				
			||||||
 | 
					                    {{
 | 
				
			||||||
 | 
					                        store.authenticated
 | 
				
			||||||
 | 
					                            ? "✅ Authenticated"
 | 
				
			||||||
 | 
					                            : "❌ Not Authenticated"
 | 
				
			||||||
 | 
					                    }}
 | 
				
			||||||
 | 
					                </p>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div class="actions">
 | 
				
			||||||
 | 
					                <button @click="store.login">Login</button>
 | 
				
			||||||
 | 
					                <button @click="store.logout">Logout</button>
 | 
				
			||||||
 | 
					                <!--<button @click="store.signup">Signup-admin</button>
 | 
				
			||||||
 | 
					                <button @click="store.signup">Signup-Entrepreneur</button>
 | 
				
			||||||
 | 
					                <button @click="store.refreshUserToken">Refresh Token</button>-->
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <!--
 | 
				
			||||||
 | 
					            <div v-if="store.authenticated" class="token-section">
 | 
				
			||||||
 | 
					                <p><strong>Access Token:</strong></p>
 | 
				
			||||||
 | 
					                <pre>{{ store.user.token }}</pre>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                <p><strong>Refresh Token:</strong></p>
 | 
				
			||||||
 | 
					                <pre>{{ store.user.refreshToken }}</pre>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div class="api-calls">
 | 
				
			||||||
 | 
					                <h2>Test API Calls</h2>
 | 
				
			||||||
 | 
					                <button @click="callApi('random')">
 | 
				
			||||||
 | 
					                    Call Entrepreneur API
 | 
				
			||||||
 | 
					                </button>
 | 
				
			||||||
 | 
					                <button @click="callApi('random2')">Call Admin API</button>
 | 
				
			||||||
 | 
					                <button @click="callApi('unauth/dev')">Call Unauth API</button>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                <div class="custom-call">
 | 
				
			||||||
 | 
					                    <input
 | 
				
			||||||
 | 
					                        v-model="customRequest"
 | 
				
			||||||
 | 
					                        placeholder="Custom endpoint"
 | 
				
			||||||
 | 
					                    />
 | 
				
			||||||
 | 
					                    <button @click="callApi(customRequest)">Call</button>
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>-->
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					.auth-container {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: center;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    padding: 3rem 1rem;
 | 
				
			||||||
 | 
					    min-height: 100vh;
 | 
				
			||||||
 | 
					    background-color: #eef1f5;
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.auth-card {
 | 
				
			||||||
 | 
					    background: white;
 | 
				
			||||||
 | 
					    padding: 2rem;
 | 
				
			||||||
 | 
					    border-radius: 12px;
 | 
				
			||||||
 | 
					    box-shadow: 0 5px 15px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    max-width: 600px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					h1 {
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.status {
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    margin-bottom: 1.5rem;
 | 
				
			||||||
 | 
					    font-weight: bold;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.success {
 | 
				
			||||||
 | 
					    color: green;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.error {
 | 
				
			||||||
 | 
					    color: red;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.actions {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-wrap: wrap;
 | 
				
			||||||
 | 
					    gap: 1rem;
 | 
				
			||||||
 | 
					    justify-content: center;
 | 
				
			||||||
 | 
					    margin-bottom: 1.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.actions button {
 | 
				
			||||||
 | 
					    padding: 0.6rem 1rem;
 | 
				
			||||||
 | 
					    background-color: #4a90e2;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    transition: background-color 0.2s;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.actions button:hover {
 | 
				
			||||||
 | 
					    background-color: #357abd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.token-section pre {
 | 
				
			||||||
 | 
					    background: #f6f8fa;
 | 
				
			||||||
 | 
					    padding: 0.5rem;
 | 
				
			||||||
 | 
					    overflow-x: auto;
 | 
				
			||||||
 | 
					    border: 1px solid #ddd;
 | 
				
			||||||
 | 
					    border-radius: 6px;
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					    font-size: 0.85rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.api-calls {
 | 
				
			||||||
 | 
					    margin-top: 2rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.api-calls h2 {
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					    color: #444;
 | 
				
			||||||
 | 
					    font-size: 1.1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.api-calls button {
 | 
				
			||||||
 | 
					    margin-right: 0.5rem;
 | 
				
			||||||
 | 
					    margin-bottom: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.custom-call {
 | 
				
			||||||
 | 
					    margin-top: 1rem;
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    gap: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.custom-call input {
 | 
				
			||||||
 | 
					    flex: 1;
 | 
				
			||||||
 | 
					    padding: 0.5rem;
 | 
				
			||||||
 | 
					    border: 1px solid #ccc;
 | 
				
			||||||
 | 
					    border-radius: 6px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					/* 
 | 
				
			||||||
 | 
					.status {
 | 
				
			||||||
 | 
					  padding: 0.5rem 1rem;
 | 
				
			||||||
 | 
					  border-radius: 8px;
 | 
				
			||||||
 | 
					  display: inline-block;
 | 
				
			||||||
 | 
					  background-color: #e0f7e9; 
 | 
				
			||||||
 | 
					  color: #2e7d32;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					.status.error {
 | 
				
			||||||
 | 
					    background-color: #ffe2e2;
 | 
				
			||||||
 | 
					    color: #c62828;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										132
									
								
								front/MyINPulse-front/src/components/PendingProjectComponent.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								front/MyINPulse-front/src/components/PendingProjectComponent.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,132 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div class="project">
 | 
				
			||||||
 | 
					        <div class="project-header">
 | 
				
			||||||
 | 
					            <div class="project-title">
 | 
				
			||||||
 | 
					                <h2>{{ projectName }}</h2>
 | 
				
			||||||
 | 
					                <p>Projet mis le: {{ creationDate }}</p>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					            <div class="project-buttons">
 | 
				
			||||||
 | 
					                <button id="accept" @click="acceptProject">Accepter</button>
 | 
				
			||||||
 | 
					                <button id="refus" @click="refuseProject">Refuser</button>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { defineProps } from "vue";
 | 
				
			||||||
 | 
					import { addNewMessage, color } from "@/services/popupDisplayer";
 | 
				
			||||||
 | 
					import { decidePendingProject } from "@/services/Apis/Admin";
 | 
				
			||||||
 | 
					import ProjectDecision from "@/ApiClasses/ProjectDecision";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const props = defineProps<{
 | 
				
			||||||
 | 
					    projectId?: number;
 | 
				
			||||||
 | 
					    projectName: string;
 | 
				
			||||||
 | 
					    creationDate: string;
 | 
				
			||||||
 | 
					    adminId?: number;
 | 
				
			||||||
 | 
					}>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const sendDecision = (isAccepted: boolean) => {
 | 
				
			||||||
 | 
					    const decision = new ProjectDecision({
 | 
				
			||||||
 | 
					        projectId: props.projectId,
 | 
				
			||||||
 | 
					        adminId: props.adminId,
 | 
				
			||||||
 | 
					        isAccepted,
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    decidePendingProject(
 | 
				
			||||||
 | 
					        decision,
 | 
				
			||||||
 | 
					        () => {
 | 
				
			||||||
 | 
					            addNewMessage(
 | 
				
			||||||
 | 
					                `Projet ${props.projectName} ${isAccepted ? "accepté" : "refusé"}`,
 | 
				
			||||||
 | 
					                color.Green
 | 
				
			||||||
 | 
					            );
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        (err) => {
 | 
				
			||||||
 | 
					            addNewMessage(`Erreur lors de la décision`, color.Red);
 | 
				
			||||||
 | 
					            console.error(err);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const acceptProject = () => sendDecision(true);
 | 
				
			||||||
 | 
					const refuseProject = () => sendDecision(false);
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					.project {
 | 
				
			||||||
 | 
					    background: linear-gradient(to right, #f8f9fb, #ffffff);
 | 
				
			||||||
 | 
					    border: 1px solid #e0e0e0;
 | 
				
			||||||
 | 
					    border-radius: 16px;
 | 
				
			||||||
 | 
					    padding: 1.5rem;
 | 
				
			||||||
 | 
					    margin: 1.5rem 0;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.05);
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					    transition: box-shadow 0.3s ease;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project:hover {
 | 
				
			||||||
 | 
					    box-shadow: 0 6px 18px rgba(0, 0, 0, 0.08);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-header {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: space-between;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    flex-wrap: wrap;
 | 
				
			||||||
 | 
					    gap: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-title {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-direction: column;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-title h2 {
 | 
				
			||||||
 | 
					    font-size: 1.25rem;
 | 
				
			||||||
 | 
					    color: #222;
 | 
				
			||||||
 | 
					    margin: 0;
 | 
				
			||||||
 | 
					    font-weight: 600;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-title p {
 | 
				
			||||||
 | 
					    font-size: 0.9rem;
 | 
				
			||||||
 | 
					    color: #666;
 | 
				
			||||||
 | 
					    margin-top: 0.25rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-buttons {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    gap: 0.75rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button {
 | 
				
			||||||
 | 
					    padding: 0.5rem 1.1rem;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    font-size: 0.9rem;
 | 
				
			||||||
 | 
					    font-weight: 500;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    transition:
 | 
				
			||||||
 | 
					        background-color 0.2s ease,
 | 
				
			||||||
 | 
					        transform 0.2s ease;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#accept {
 | 
				
			||||||
 | 
					    background-color: #4caf50;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#accept:hover {
 | 
				
			||||||
 | 
					    background-color: #3e8e41;
 | 
				
			||||||
 | 
					    transform: translateY(-2px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#refus {
 | 
				
			||||||
 | 
					    background-color: #e74c3c;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#refus:hover {
 | 
				
			||||||
 | 
					    background-color: #c0392b;
 | 
				
			||||||
 | 
					    transform: translateY(-2px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										228
									
								
								front/MyINPulse-front/src/components/PendingRequestsManager.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										228
									
								
								front/MyINPulse-front/src/components/PendingRequestsManager.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,228 @@
 | 
				
			|||||||
 | 
					<!-- <template>
 | 
				
			||||||
 | 
					    <section class="pending-requests">
 | 
				
			||||||
 | 
					      <h2>Comptes en attente</h2>
 | 
				
			||||||
 | 
					      <ul v-if="pendingAccounts.length">
 | 
				
			||||||
 | 
					        <li v-for="account in pendingAccounts" :key="account.userId">
 | 
				
			||||||
 | 
					          {{ account.userName }} ({{ account.email }})
 | 
				
			||||||
 | 
					          <button @click="validateAccount(account.userId)">Valider</button>
 | 
				
			||||||
 | 
					        </li>
 | 
				
			||||||
 | 
					      </ul>
 | 
				
			||||||
 | 
					      <p v-else>Aucun compte en attente</p>
 | 
				
			||||||
 | 
					  
 | 
				
			||||||
 | 
					      <h2>Demandes de participation aux projets</h2>
 | 
				
			||||||
 | 
					      <ul v-if="joinRequests.length">
 | 
				
			||||||
 | 
					        <li v-for="request in joinRequests" :key="request.joinRequestId">
 | 
				
			||||||
 | 
					          {{ request.userName }} veut rejoindre le projet {{ request.projectName }}
 | 
				
			||||||
 | 
					          <button @click="decideJoinRequest(request.joinRequestId, true)">Accepter</button>
 | 
				
			||||||
 | 
					          <button @click="decideJoinRequest(request.joinRequestId, false)">Refuser</button>
 | 
				
			||||||
 | 
					        </li>
 | 
				
			||||||
 | 
					      </ul>
 | 
				
			||||||
 | 
					      <p v-else>Aucune demande de participation</p>
 | 
				
			||||||
 | 
					    </section>
 | 
				
			||||||
 | 
					  </template> -->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <section class="section">
 | 
				
			||||||
 | 
					        <h2>Comptes en attente</h2>
 | 
				
			||||||
 | 
					        <div v-if="pendingAccounts.length">
 | 
				
			||||||
 | 
					            <div
 | 
				
			||||||
 | 
					                v-for="account in pendingAccounts"
 | 
				
			||||||
 | 
					                :key="account.userId"
 | 
				
			||||||
 | 
					                class="request-item"
 | 
				
			||||||
 | 
					            >
 | 
				
			||||||
 | 
					                <div class="request-info">
 | 
				
			||||||
 | 
					                    {{ account.userName }} ({{ account.email }})
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					                <div class="request-buttons">
 | 
				
			||||||
 | 
					                    <button
 | 
				
			||||||
 | 
					                        class="accept"
 | 
				
			||||||
 | 
					                        @click="validateAccount(account.userId)"
 | 
				
			||||||
 | 
					                    >
 | 
				
			||||||
 | 
					                        Valider
 | 
				
			||||||
 | 
					                    </button>
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					        <div v-else class="empty-message">Aucun compte en attente</div>
 | 
				
			||||||
 | 
					    </section>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <section class="section">
 | 
				
			||||||
 | 
					        <h2>Demandes de participation aux projets</h2>
 | 
				
			||||||
 | 
					        <div v-if="joinRequests.length">
 | 
				
			||||||
 | 
					            <div
 | 
				
			||||||
 | 
					                v-for="request in joinRequests"
 | 
				
			||||||
 | 
					                :key="request.joinRequestId"
 | 
				
			||||||
 | 
					                class="request-item"
 | 
				
			||||||
 | 
					            >
 | 
				
			||||||
 | 
					                <div class="request-info">
 | 
				
			||||||
 | 
					                    {{ request.userName }} veut rejoindre le projet "{{
 | 
				
			||||||
 | 
					                        request.projectName
 | 
				
			||||||
 | 
					                    }}"
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					                <div class="request-buttons">
 | 
				
			||||||
 | 
					                    <button
 | 
				
			||||||
 | 
					                        class="accept"
 | 
				
			||||||
 | 
					                        @click="decideJoinRequest(request.joinRequestId, true)"
 | 
				
			||||||
 | 
					                    >
 | 
				
			||||||
 | 
					                        Accepter
 | 
				
			||||||
 | 
					                    </button>
 | 
				
			||||||
 | 
					                    <button
 | 
				
			||||||
 | 
					                        class="reject"
 | 
				
			||||||
 | 
					                        @click="decideJoinRequest(request.joinRequestId, false)"
 | 
				
			||||||
 | 
					                    >
 | 
				
			||||||
 | 
					                        Refuser
 | 
				
			||||||
 | 
					                    </button>
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					        <div v-else class="empty-message">Aucune demande de participation</div>
 | 
				
			||||||
 | 
					    </section>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { ref, onMounted } from "vue";
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					    getPendingAccounts,
 | 
				
			||||||
 | 
					    validateUserAccount,
 | 
				
			||||||
 | 
					    getPendingProjectJoinRequests,
 | 
				
			||||||
 | 
					    decideProjectJoinRequest,
 | 
				
			||||||
 | 
					} from "@/services/Apis/Admin";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type PendingAccount = {
 | 
				
			||||||
 | 
					    userId: number;
 | 
				
			||||||
 | 
					    userName: string;
 | 
				
			||||||
 | 
					    email: string;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type JoinRequest = {
 | 
				
			||||||
 | 
					    joinRequestId: number;
 | 
				
			||||||
 | 
					    userName: string;
 | 
				
			||||||
 | 
					    projectName: string;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const pendingAccounts = ref<PendingAccount[]>([]);
 | 
				
			||||||
 | 
					const joinRequests = ref<JoinRequest[]>([]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function fetchAccounts() {
 | 
				
			||||||
 | 
					    getPendingAccounts(
 | 
				
			||||||
 | 
					        (res) => (pendingAccounts.value = res.data),
 | 
				
			||||||
 | 
					        (err) => {
 | 
				
			||||||
 | 
					            console.error("Erreur lors du chargement des comptes", err);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function fetchJoinRequests() {
 | 
				
			||||||
 | 
					    getPendingProjectJoinRequests(
 | 
				
			||||||
 | 
					        (res) => (joinRequests.value = res.data),
 | 
				
			||||||
 | 
					        (err) => {
 | 
				
			||||||
 | 
					            console.error("Erreur lors du chargement des demandes", err);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function validateAccount(userId: number) {
 | 
				
			||||||
 | 
					    validateUserAccount(userId, () => {
 | 
				
			||||||
 | 
					        pendingAccounts.value = pendingAccounts.value.filter(
 | 
				
			||||||
 | 
					            (a) => a.userId !== userId
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function decideJoinRequest(joinRequestId: number, isAccepted: boolean) {
 | 
				
			||||||
 | 
					    decideProjectJoinRequest(joinRequestId, { isAccepted }, () => {
 | 
				
			||||||
 | 
					        joinRequests.value = joinRequests.value.filter(
 | 
				
			||||||
 | 
					            (r) => r.joinRequestId !== joinRequestId
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onMounted(() => {
 | 
				
			||||||
 | 
					    fetchAccounts();
 | 
				
			||||||
 | 
					    fetchJoinRequests();
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style>
 | 
				
			||||||
 | 
					.section {
 | 
				
			||||||
 | 
					    background: #ffffff;
 | 
				
			||||||
 | 
					    border-radius: 16px;
 | 
				
			||||||
 | 
					    padding: 1.5rem;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.05);
 | 
				
			||||||
 | 
					    margin-bottom: 2rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.section h2 {
 | 
				
			||||||
 | 
					    font-size: 1.4rem;
 | 
				
			||||||
 | 
					    color: #2c3e50;
 | 
				
			||||||
 | 
					    border-bottom: 2px solid #ddd;
 | 
				
			||||||
 | 
					    padding-bottom: 0.5rem;
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.request-item {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: space-between;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    border: 1px solid #e0e0e0;
 | 
				
			||||||
 | 
					    border-radius: 12px;
 | 
				
			||||||
 | 
					    background: linear-gradient(to right, #f8f9fb, #ffffff);
 | 
				
			||||||
 | 
					    padding: 1rem 1.25rem;
 | 
				
			||||||
 | 
					    margin-bottom: 1rem;
 | 
				
			||||||
 | 
					    transition: box-shadow 0.2s ease;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.request-item:hover {
 | 
				
			||||||
 | 
					    box-shadow: 0 6px 16px rgba(0, 0, 0, 0.06);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.request-info {
 | 
				
			||||||
 | 
					    flex: 1;
 | 
				
			||||||
 | 
					    font-size: 1rem;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.request-buttons {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    gap: 0.75rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.request-buttons button {
 | 
				
			||||||
 | 
					    padding: 0.45rem 1rem;
 | 
				
			||||||
 | 
					    font-size: 0.9rem;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    color: #fff;
 | 
				
			||||||
 | 
					    font-weight: 500;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    transition:
 | 
				
			||||||
 | 
					        background-color 0.2s ease,
 | 
				
			||||||
 | 
					        transform 0.2s ease;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.request-buttons button:hover {
 | 
				
			||||||
 | 
					    transform: translateY(-1px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button.accept {
 | 
				
			||||||
 | 
					    background-color: #4caf50;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button.accept:hover {
 | 
				
			||||||
 | 
					    background-color: #3e8e41;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button.reject {
 | 
				
			||||||
 | 
					    background-color: #e74c3c;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					button.reject:hover {
 | 
				
			||||||
 | 
					    background-color: #c0392b;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.empty-message {
 | 
				
			||||||
 | 
					    font-style: italic;
 | 
				
			||||||
 | 
					    color: #777;
 | 
				
			||||||
 | 
					    padding: 1rem 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
@@ -1,21 +1,306 @@
 | 
				
			|||||||
<template>
 | 
					<template>
 | 
				
			||||||
    <div class="project">
 | 
					    <div class="project">
 | 
				
			||||||
        <div class="project-header">
 | 
					        <div class="project-header">
 | 
				
			||||||
            <h2>{{ projectName }}</h2>
 | 
					            <h2 @click="goToLink">{{ projectName }}</h2>
 | 
				
			||||||
 | 
					            <div class="header-actions">
 | 
				
			||||||
 | 
					                <div ref="dropdownRef" class="dropdown-wrapper">
 | 
				
			||||||
 | 
					                    <button class="contact-button" @click.stop="toggleDropdown">
 | 
				
			||||||
 | 
					                        Contact
 | 
				
			||||||
 | 
					                    </button>
 | 
				
			||||||
 | 
					                    <div
 | 
				
			||||||
 | 
					                        v-if="entrepreneurEmails.length > 0"
 | 
				
			||||||
 | 
					                        class="contact-dropdown"
 | 
				
			||||||
 | 
					                        :class="{ 'dropdown-visible': isDropdownOpen }"
 | 
				
			||||||
 | 
					                    >
 | 
				
			||||||
 | 
					                        <button @click="contactAll">Contacter tous</button>
 | 
				
			||||||
 | 
					                        <button
 | 
				
			||||||
 | 
					                            v-for="(email, index) in entrepreneurEmails"
 | 
				
			||||||
 | 
					                            :key="index"
 | 
				
			||||||
 | 
					                            @click="contactSingle(email)"
 | 
				
			||||||
 | 
					                        >
 | 
				
			||||||
 | 
					                            {{ email }}
 | 
				
			||||||
 | 
					                        </button>
 | 
				
			||||||
 | 
					                    </div>
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <!-- Toute cette partie est cliquable -->
 | 
				
			||||||
 | 
					        <div class="project-body" @click="goToLink">
 | 
				
			||||||
 | 
					            <ul>
 | 
				
			||||||
 | 
					                <li v-for="(name, index) in listName" :key="index">
 | 
				
			||||||
 | 
					                    {{ name }}
 | 
				
			||||||
 | 
					                </li>
 | 
				
			||||||
 | 
					            </ul>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<script lang="ts">
 | 
					<script setup lang="ts">
 | 
				
			||||||
import type { PropType } from "vue";
 | 
					import { defineProps, ref, onMounted, onBeforeUnmount } from "vue";
 | 
				
			||||||
 | 
					import { useRouter } from "vue-router";
 | 
				
			||||||
 | 
					import { getProjectEntrepreneurs } from "@/services/Apis/Shared.ts";
 | 
				
			||||||
 | 
					import UserEntrepreneur from "@/ApiClasses/UserEntrepreneur.ts";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default {
 | 
					const IS_MOCK_MODE = false;
 | 
				
			||||||
    name: "ProjectComponent",
 | 
					const dropdownRef = ref<HTMLElement | null>(null);
 | 
				
			||||||
    props: {
 | 
					
 | 
				
			||||||
        projectName: {
 | 
					const props = defineProps<{
 | 
				
			||||||
            type: Object as PropType<string>,
 | 
					    projectName: string;
 | 
				
			||||||
            required: true,
 | 
					    listName: string[];
 | 
				
			||||||
        },
 | 
					    projectLink: string;
 | 
				
			||||||
    },
 | 
					    projectId: number;
 | 
				
			||||||
 | 
					}>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const router = useRouter();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const isDropdownOpen = ref(false);
 | 
				
			||||||
 | 
					const entrepreneurEmails = ref<string[]>([]);
 | 
				
			||||||
 | 
					const entrepreneurs = ref<UserEntrepreneur[]>([]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const goToLink = () => {
 | 
				
			||||||
 | 
					    if (props.projectLink) {
 | 
				
			||||||
 | 
					        router.push(props.projectLink);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const toggleDropdown = () => {
 | 
				
			||||||
 | 
					    isDropdownOpen.value = !isDropdownOpen.value;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const fetchMockEntrepreneurs = () => {
 | 
				
			||||||
 | 
					    const mockData = [
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            userName: "Doe",
 | 
				
			||||||
 | 
					            userSurname: "John",
 | 
				
			||||||
 | 
					            primaryMail: "john.doe@example.com",
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            userName: "Smith",
 | 
				
			||||||
 | 
					            userSurname: "Anna",
 | 
				
			||||||
 | 
					            primaryMail: "anna.smith@example.com",
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            userName: "Mock",
 | 
				
			||||||
 | 
					            userSurname: "User",
 | 
				
			||||||
 | 
					            primaryMail: undefined,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    entrepreneurs.value = mockData.map((item) => new UserEntrepreneur(item));
 | 
				
			||||||
 | 
					    entrepreneurEmails.value = entrepreneurs.value
 | 
				
			||||||
 | 
					        .map((e) => e.primaryMail)
 | 
				
			||||||
 | 
					        .filter((mail): mail is string => !!mail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    console.log("Mock entrepreneurs chargés :", entrepreneurs.value);
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const fetchEntrepreneurs = (projectId: number, useMock = false) => {
 | 
				
			||||||
 | 
					    if (useMock) {
 | 
				
			||||||
 | 
					        fetchMockEntrepreneurs();
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        getProjectEntrepreneurs(
 | 
				
			||||||
 | 
					            projectId,
 | 
				
			||||||
 | 
					            (response) => {
 | 
				
			||||||
 | 
					                const rawData = response.data as Partial<UserEntrepreneur>[];
 | 
				
			||||||
 | 
					                entrepreneurs.value = rawData.map(
 | 
				
			||||||
 | 
					                    (item) => new UserEntrepreneur(item)
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					                entrepreneurEmails.value = entrepreneurs.value
 | 
				
			||||||
 | 
					                    .map((e) => e.primaryMail)
 | 
				
			||||||
 | 
					                    .filter((mail): mail is string => !!mail);
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            (error) => {
 | 
				
			||||||
 | 
					                console.error(
 | 
				
			||||||
 | 
					                    "Erreur lors de la récupération des entrepreneurs :",
 | 
				
			||||||
 | 
					                    error
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const contactAll = () => {
 | 
				
			||||||
 | 
					    const allEmails = entrepreneurEmails.value.join(", ");
 | 
				
			||||||
 | 
					    navigator.clipboard
 | 
				
			||||||
 | 
					        .writeText(allEmails)
 | 
				
			||||||
 | 
					        .then(() => {
 | 
				
			||||||
 | 
					            alert("Tous les emails copiés dans le presse-papiers !");
 | 
				
			||||||
 | 
					            window.open("https://partage.bordeaux-inp.fr/", "_blank");
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((err) => console.error("Erreur lors de la copie :", err));
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const contactSingle = (email: string) => {
 | 
				
			||||||
 | 
					    navigator.clipboard
 | 
				
			||||||
 | 
					        .writeText(email)
 | 
				
			||||||
 | 
					        .then(() => {
 | 
				
			||||||
 | 
					            alert(`Adresse copiée : ${email}`);
 | 
				
			||||||
 | 
					            window.open("https://partage.bordeaux-inp.fr/", "_blank");
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((err) => console.error("Erreur lors de la copie :", err));
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const handleClickOutside = (event: MouseEvent) => {
 | 
				
			||||||
 | 
					    if (
 | 
				
			||||||
 | 
					        isDropdownOpen.value &&
 | 
				
			||||||
 | 
					        dropdownRef.value &&
 | 
				
			||||||
 | 
					        !dropdownRef.value.contains(event.target as Node)
 | 
				
			||||||
 | 
					    ) {
 | 
				
			||||||
 | 
					        isDropdownOpen.value = false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onMounted(() => {
 | 
				
			||||||
 | 
					    fetchEntrepreneurs(props.projectId, IS_MOCK_MODE);
 | 
				
			||||||
 | 
					    document.addEventListener("click", handleClickOutside);
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onBeforeUnmount(() => {
 | 
				
			||||||
 | 
					    document.removeEventListener("click", handleClickOutside);
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					.project {
 | 
				
			||||||
 | 
					    background: linear-gradient(to right, #f8f9fb, #ffffff);
 | 
				
			||||||
 | 
					    border: 1px solid #e0e0e0;
 | 
				
			||||||
 | 
					    border-radius: 16px;
 | 
				
			||||||
 | 
					    padding: 1.5rem;
 | 
				
			||||||
 | 
					    margin: 1.5rem 0;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.05);
 | 
				
			||||||
 | 
					    transition: box-shadow 0.3s ease;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project:hover {
 | 
				
			||||||
 | 
					    box-shadow: 0 6px 18px rgba(0, 0, 0, 0.08);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-header {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: space-between;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    flex-wrap: wrap;
 | 
				
			||||||
 | 
					    gap: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-header h2 {
 | 
				
			||||||
 | 
					    font-size: 1.25rem;
 | 
				
			||||||
 | 
					    color: #222;
 | 
				
			||||||
 | 
					    margin: 0;
 | 
				
			||||||
 | 
					    font-weight: 600;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-buttons {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    gap: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-btn {
 | 
				
			||||||
 | 
					    background-color: #007bff;
 | 
				
			||||||
 | 
					    color: #fff;
 | 
				
			||||||
 | 
					    padding: 0.5rem 1rem;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    font-size: 0.9rem;
 | 
				
			||||||
 | 
					    font-weight: 500;
 | 
				
			||||||
 | 
					    transition:
 | 
				
			||||||
 | 
					        background-color 0.2s ease,
 | 
				
			||||||
 | 
					        transform 0.2s ease;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-btn:hover {
 | 
				
			||||||
 | 
					    background-color: #0056b3;
 | 
				
			||||||
 | 
					    transform: translateY(-2px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-body {
 | 
				
			||||||
 | 
					    margin-top: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-body ul {
 | 
				
			||||||
 | 
					    list-style-type: disc;
 | 
				
			||||||
 | 
					    padding-left: 1.25rem;
 | 
				
			||||||
 | 
					    margin: 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.project-body ul li {
 | 
				
			||||||
 | 
					    font-size: 0.95rem;
 | 
				
			||||||
 | 
					    color: #555;
 | 
				
			||||||
 | 
					    line-height: 1.6;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.header {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: space-between;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    padding: 15px 30px;
 | 
				
			||||||
 | 
					    background-color: #f9f9f9;
 | 
				
			||||||
 | 
					    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.logo {
 | 
				
			||||||
 | 
					    height: 50px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.header-actions {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    gap: 20px;
 | 
				
			||||||
 | 
					    position: relative;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-button,
 | 
				
			||||||
 | 
					.return-button {
 | 
				
			||||||
 | 
					    background-color: #009cde;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    padding: 10px 15px;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    font-size: 14px;
 | 
				
			||||||
 | 
					    border-radius: 5px;
 | 
				
			||||||
 | 
					    text-decoration: none;
 | 
				
			||||||
 | 
					    transition: background-color 0.2s ease;
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.return-button:hover,
 | 
				
			||||||
 | 
					.contact-button:hover {
 | 
				
			||||||
 | 
					    background-color: #007bad;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown {
 | 
				
			||||||
 | 
					    position: absolute;
 | 
				
			||||||
 | 
					    top: 100%;
 | 
				
			||||||
 | 
					    left: 0;
 | 
				
			||||||
 | 
					    background-color: #000;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    box-shadow: 0px 4px 8px rgba(255, 255, 255, 0.2);
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					    margin-top: 5px;
 | 
				
			||||||
 | 
					    z-index: 1000;
 | 
				
			||||||
 | 
					    min-width: 200px;
 | 
				
			||||||
 | 
					    display: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown button {
 | 
				
			||||||
 | 
					    display: block;
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    padding: 5px;
 | 
				
			||||||
 | 
					    text-align: left;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    background: none;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown button:hover {
 | 
				
			||||||
 | 
					    background-color: #009cde;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown.dropdown-visible {
 | 
				
			||||||
 | 
					    display: block;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										687
									
								
								front/MyINPulse-front/src/components/canvas/CanvasItem.vue
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										687
									
								
								front/MyINPulse-front/src/components/canvas/CanvasItem.vue
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,687 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div :class="['cell', { expanded }]" @click="handleClick">
 | 
				
			||||||
 | 
					        <h3 class="fs-5 fw-medium">{{ titleText }}</h3>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <div class="tooltip-explain">{{ description }}</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <template v-if="expanded">
 | 
				
			||||||
 | 
					            <div class="explain">
 | 
				
			||||||
 | 
					                <p>{{ description }}</p>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					        </template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <div class="description-wrapper custom-flow">
 | 
				
			||||||
 | 
					            <div
 | 
				
			||||||
 | 
					                v-for="(desc, index) in currentDescriptions"
 | 
				
			||||||
 | 
					                :key="desc.idSectionCell || index"
 | 
				
			||||||
 | 
					                :class="[
 | 
				
			||||||
 | 
					                    'section-bloc',
 | 
				
			||||||
 | 
					                    index % 2 === 0 ? 'from-left' : 'from-right',
 | 
				
			||||||
 | 
					                ]"
 | 
				
			||||||
 | 
					            >
 | 
				
			||||||
 | 
					                <!-- ADMIN -------------------------------------------------------------------------------------------->
 | 
				
			||||||
 | 
					                <template v-if="IS_ADMIN">
 | 
				
			||||||
 | 
					                    <div class="description">
 | 
				
			||||||
 | 
					                        <p class="m-0">{{ desc.contentSectionCell }}</p>
 | 
				
			||||||
 | 
					                    </div>
 | 
				
			||||||
 | 
					                </template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                <!-- ENTREP ------------------------------------------------------------------------------------------->
 | 
				
			||||||
 | 
					                <template v-else>
 | 
				
			||||||
 | 
					                    <!-- Mode affichage -->
 | 
				
			||||||
 | 
					                    <template v-if="!isEditing[index]">
 | 
				
			||||||
 | 
					                        <template v-if="expanded">
 | 
				
			||||||
 | 
					                            <button
 | 
				
			||||||
 | 
					                                class="delete-button"
 | 
				
			||||||
 | 
					                                title="Supprimer"
 | 
				
			||||||
 | 
					                                @click.stop="
 | 
				
			||||||
 | 
					                                    deleteSectionCell(desc.idSectionCell, index)
 | 
				
			||||||
 | 
					                                "
 | 
				
			||||||
 | 
					                            >
 | 
				
			||||||
 | 
					                                ✕
 | 
				
			||||||
 | 
					                            </button>
 | 
				
			||||||
 | 
					                        </template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                        <div class="description">
 | 
				
			||||||
 | 
					                            <p class="m-0">{{ desc.contentSectionCell }}</p>
 | 
				
			||||||
 | 
					                        </div>
 | 
				
			||||||
 | 
					                        <div class="button-container">
 | 
				
			||||||
 | 
					                            <button
 | 
				
			||||||
 | 
					                                v-if="expanded"
 | 
				
			||||||
 | 
					                                class="edit-button"
 | 
				
			||||||
 | 
					                                @click.stop="startEditing(index)"
 | 
				
			||||||
 | 
					                            >
 | 
				
			||||||
 | 
					                                Éditer
 | 
				
			||||||
 | 
					                            </button>
 | 
				
			||||||
 | 
					                        </div>
 | 
				
			||||||
 | 
					                    </template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    <!-- Mode édition -->
 | 
				
			||||||
 | 
					                    <template v-else>
 | 
				
			||||||
 | 
					                        <div class="edit-row">
 | 
				
			||||||
 | 
					                            <textarea
 | 
				
			||||||
 | 
					                                v-model="
 | 
				
			||||||
 | 
					                                    editedDescriptions[index].contentSectionCell
 | 
				
			||||||
 | 
					                                "
 | 
				
			||||||
 | 
					                                class="edit-input"
 | 
				
			||||||
 | 
					                            ></textarea>
 | 
				
			||||||
 | 
					                            <div class="button-container">
 | 
				
			||||||
 | 
					                                <button
 | 
				
			||||||
 | 
					                                    class="save-button"
 | 
				
			||||||
 | 
					                                    @click.stop="saveEdit(index)"
 | 
				
			||||||
 | 
					                                >
 | 
				
			||||||
 | 
					                                    Enregistrer
 | 
				
			||||||
 | 
					                                </button>
 | 
				
			||||||
 | 
					                                <button
 | 
				
			||||||
 | 
					                                    class="cancel-button"
 | 
				
			||||||
 | 
					                                    @click.stop="cancelEdit(index)"
 | 
				
			||||||
 | 
					                                >
 | 
				
			||||||
 | 
					                                    Annuler
 | 
				
			||||||
 | 
					                                </button>
 | 
				
			||||||
 | 
					                            </div>
 | 
				
			||||||
 | 
					                        </div>
 | 
				
			||||||
 | 
					                    </template>
 | 
				
			||||||
 | 
					                </template>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <template v-if="expanded">
 | 
				
			||||||
 | 
					                <div class="canvas-exit-hint">
 | 
				
			||||||
 | 
					                    Cliquez n'importe où pour quitter le canvas (terminez
 | 
				
			||||||
 | 
					                    d'abord vos modifications)
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </template>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { ref, defineProps, onMounted } from "vue";
 | 
				
			||||||
 | 
					import { getSectionCellsByDate } from "@/services/Apis/Shared.ts";
 | 
				
			||||||
 | 
					import { addSectionCell } from "@/services/Apis/Entrepreneurs.ts";
 | 
				
			||||||
 | 
					import SectionCell from "@/ApiClasses/SectionCell";
 | 
				
			||||||
 | 
					import { removeSectionCell } from "@/services/Apis/Entrepreneurs.ts";
 | 
				
			||||||
 | 
					import type { AxiosResponse, AxiosError } from "axios";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const props = defineProps<{
 | 
				
			||||||
 | 
					    projectId: number;
 | 
				
			||||||
 | 
					    title: number;
 | 
				
			||||||
 | 
					    titleText: string;
 | 
				
			||||||
 | 
					    description: string;
 | 
				
			||||||
 | 
					    isAdmin: boolean;
 | 
				
			||||||
 | 
					}>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const IS_MOCK_MODE = true;
 | 
				
			||||||
 | 
					const IS_ADMIN = props.isAdmin;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const expanded = ref(false);
 | 
				
			||||||
 | 
					const currentDescriptions = ref<SectionCell[]>([]);
 | 
				
			||||||
 | 
					const editedDescriptions = ref<SectionCell[]>([]);
 | 
				
			||||||
 | 
					const isEditing = ref<boolean[]>([]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const startEditing = (index: number) => {
 | 
				
			||||||
 | 
					    isEditing.value[index] = true;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const cancelEdit = (index: number) => {
 | 
				
			||||||
 | 
					    editedDescriptions.value[index].contentSectionCell =
 | 
				
			||||||
 | 
					        currentDescriptions.value[index].contentSectionCell;
 | 
				
			||||||
 | 
					    isEditing.value[index] = false;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const saveEdit = (index: number) => {
 | 
				
			||||||
 | 
					    currentDescriptions.value[index].contentSectionCell =
 | 
				
			||||||
 | 
					        editedDescriptions.value[index].contentSectionCell;
 | 
				
			||||||
 | 
					    isEditing.value[index] = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (!IS_MOCK_MODE) {
 | 
				
			||||||
 | 
					        addSectionCell(
 | 
				
			||||||
 | 
					            currentDescriptions.value[index],
 | 
				
			||||||
 | 
					            (response) => {
 | 
				
			||||||
 | 
					                console.log(
 | 
				
			||||||
 | 
					                    "Modification enregistrée avec succès :",
 | 
				
			||||||
 | 
					                    response.data
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            (error) => {
 | 
				
			||||||
 | 
					                console.error("Erreur lors de l'enregistrement :", error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const deleteSectionCell = (id: number | undefined, index: number): void => {
 | 
				
			||||||
 | 
					    if (id === -1) {
 | 
				
			||||||
 | 
					        window.confirm("Êtes-vous sûr de vouloir supprimer cet élément ?");
 | 
				
			||||||
 | 
					        console.error("Delete ignored");
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const confirmed = window.confirm(
 | 
				
			||||||
 | 
					        "Êtes-vous sûr de vouloir supprimer cet élément ?"
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (!confirmed) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (id === undefined) {
 | 
				
			||||||
 | 
					        console.error("ID de la cellule non défini");
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    removeSectionCell(
 | 
				
			||||||
 | 
					        id,
 | 
				
			||||||
 | 
					        () => {
 | 
				
			||||||
 | 
					            currentDescriptions.value.splice(index, 1);
 | 
				
			||||||
 | 
					            editedDescriptions.value.splice(index, 1);
 | 
				
			||||||
 | 
					            isEditing.value.splice(index, 1);
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        (error: AxiosError) => {
 | 
				
			||||||
 | 
					            console.error("Erreur lors de la suppression :", error);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const handleClick = () => {
 | 
				
			||||||
 | 
					    if (expanded.value) {
 | 
				
			||||||
 | 
					        const editingInProgress = isEditing.value.some((edit) => edit);
 | 
				
			||||||
 | 
					        if (!editingInProgress) {
 | 
				
			||||||
 | 
					            expanded.value = false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        expanded.value = true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const handleFetchSuccess = (sectionCells: SectionCell[]) => {
 | 
				
			||||||
 | 
					    currentDescriptions.value = sectionCells;
 | 
				
			||||||
 | 
					    editedDescriptions.value = sectionCells.map(
 | 
				
			||||||
 | 
					        (cell) =>
 | 
				
			||||||
 | 
					            new SectionCell({
 | 
				
			||||||
 | 
					                idSectionCell: cell.idSectionCell,
 | 
				
			||||||
 | 
					                sectionId: cell.sectionId,
 | 
				
			||||||
 | 
					                contentSectionCell: cell.contentSectionCell,
 | 
				
			||||||
 | 
					                modificationDate: cell.modificationDate,
 | 
				
			||||||
 | 
					            })
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					    isEditing.value = Array(sectionCells.length).fill(false);
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const handleFetchError = (error: unknown) => {
 | 
				
			||||||
 | 
					    console.error("Erreur lors de la récupération des données :", error);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const errorCell = new SectionCell({
 | 
				
			||||||
 | 
					        idSectionCell: -1,
 | 
				
			||||||
 | 
					        sectionId: -1,
 | 
				
			||||||
 | 
					        contentSectionCell: "Échec du chargement des données.",
 | 
				
			||||||
 | 
					        modificationDate: new Date().toISOString(),
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    currentDescriptions.value = [errorCell];
 | 
				
			||||||
 | 
					    editedDescriptions.value = [errorCell];
 | 
				
			||||||
 | 
					    isEditing.value = [false];
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const fetchData = async (
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    title: number,
 | 
				
			||||||
 | 
					    date: string,
 | 
				
			||||||
 | 
					    useMock = false
 | 
				
			||||||
 | 
					) => {
 | 
				
			||||||
 | 
					    try {
 | 
				
			||||||
 | 
					        if (useMock) {
 | 
				
			||||||
 | 
					            const responseData = await mockFetch(projectId, title, date);
 | 
				
			||||||
 | 
					            handleFetchSuccess(responseData);
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            if (projectId == -1) {
 | 
				
			||||||
 | 
					                const errorCell = new SectionCell({
 | 
				
			||||||
 | 
					                    idSectionCell: -1,
 | 
				
			||||||
 | 
					                    sectionId: -1,
 | 
				
			||||||
 | 
					                    contentSectionCell: "Échec du chargement des données.",
 | 
				
			||||||
 | 
					                    modificationDate: new Date().toISOString(),
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                currentDescriptions.value = [errorCell];
 | 
				
			||||||
 | 
					                editedDescriptions.value = [errorCell];
 | 
				
			||||||
 | 
					                isEditing.value = [false];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                console.error(
 | 
				
			||||||
 | 
					                    "No sections to show because no project was found."
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					                return;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            await new Promise<void>((resolve, reject) => {
 | 
				
			||||||
 | 
					                getSectionCellsByDate(
 | 
				
			||||||
 | 
					                    projectId,
 | 
				
			||||||
 | 
					                    title,
 | 
				
			||||||
 | 
					                    date,
 | 
				
			||||||
 | 
					                    (response: AxiosResponse) => {
 | 
				
			||||||
 | 
					                        const data = response.data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                        if (Array.isArray(data) && data.length > 0) {
 | 
				
			||||||
 | 
					                            const sectionCells = data.map(
 | 
				
			||||||
 | 
					                                (cellData) =>
 | 
				
			||||||
 | 
					                                    new SectionCell({
 | 
				
			||||||
 | 
					                                        idSectionCell: cellData.idSectionCell,
 | 
				
			||||||
 | 
					                                        sectionId: cellData.sectionId,
 | 
				
			||||||
 | 
					                                        contentSectionCell:
 | 
				
			||||||
 | 
					                                            cellData.contentSectionCell,
 | 
				
			||||||
 | 
					                                        modificationDate:
 | 
				
			||||||
 | 
					                                            cellData.modificationDate,
 | 
				
			||||||
 | 
					                                    })
 | 
				
			||||||
 | 
					                            );
 | 
				
			||||||
 | 
					                            handleFetchSuccess(sectionCells);
 | 
				
			||||||
 | 
					                        } else {
 | 
				
			||||||
 | 
					                            console.warn(
 | 
				
			||||||
 | 
					                                "Aucune donnée reçue ou format inattendu :",
 | 
				
			||||||
 | 
					                                data
 | 
				
			||||||
 | 
					                            );
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                        resolve();
 | 
				
			||||||
 | 
					                    },
 | 
				
			||||||
 | 
					                    (error: AxiosError) => {
 | 
				
			||||||
 | 
					                        handleFetchError(error);
 | 
				
			||||||
 | 
					                        reject(error);
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    } catch (error) {
 | 
				
			||||||
 | 
					        handleFetchError(error);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const mockFetch = async (
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    title: number,
 | 
				
			||||||
 | 
					    date: string
 | 
				
			||||||
 | 
					): Promise<SectionCell[]> => {
 | 
				
			||||||
 | 
					    console.log(
 | 
				
			||||||
 | 
					        `Mock fetch pour projectId: ${projectId}, title: ${title}, date: ${date}`
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const leanCanvasData: Record<number, string[]> = {
 | 
				
			||||||
 | 
					        1: [
 | 
				
			||||||
 | 
					            "Les clients ont du mal à trouver des produits écoresponsables abordables.",
 | 
				
			||||||
 | 
					            "Le processus d'achat en ligne est trop complexe.",
 | 
				
			||||||
 | 
					            "Manque de transparence sur l’origine des produits.",
 | 
				
			||||||
 | 
					            "Peu d’alternatives locales et durables sur le marché.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        2: [
 | 
				
			||||||
 | 
					            "Jeunes urbains engagés dans la cause écologique.",
 | 
				
			||||||
 | 
					            "Familles à revenu moyen voulant consommer responsable.",
 | 
				
			||||||
 | 
					            "Entreprises soucieuses de leur empreinte carbone.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        3: [
 | 
				
			||||||
 | 
					            "Une plateforme centralisée avec des produits écologiques certifiés.",
 | 
				
			||||||
 | 
					            "Un service client humain et réactif.",
 | 
				
			||||||
 | 
					            "Livraison éco-responsable avec suivi.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        4: [
 | 
				
			||||||
 | 
					            "Application intuitive avec suggestions personnalisées.",
 | 
				
			||||||
 | 
					            "Emballages recyclables et réutilisables.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        5: [
 | 
				
			||||||
 | 
					            "Algorithme exclusif de recommandations durables.",
 | 
				
			||||||
 | 
					            "Forte communauté engagée sur les réseaux.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        6: [
 | 
				
			||||||
 | 
					            "Canaux digitaux : réseaux sociaux, SEO.",
 | 
				
			||||||
 | 
					            "Partenariats avec influenceurs écoresponsables.",
 | 
				
			||||||
 | 
					            "Boutique physique en pop-up stores.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        7: [
 | 
				
			||||||
 | 
					            "Taux de rétention client mensuel.",
 | 
				
			||||||
 | 
					            "Taux de satisfaction utilisateur (NPS).",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        8: [
 | 
				
			||||||
 | 
					            "Coût du développement logiciel initial.",
 | 
				
			||||||
 | 
					            "Campagnes publicitaires et communication.",
 | 
				
			||||||
 | 
					            "Frais logistiques (emballages, transport).",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        9: [
 | 
				
			||||||
 | 
					            "Ventes directes sur la plateforme.",
 | 
				
			||||||
 | 
					            "Abonnement mensuel premium pour livraison gratuite.",
 | 
				
			||||||
 | 
					            "Revenus via partenariats de marque.",
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const section = leanCanvasData[title] || ["Aucune donnée disponible."];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const result = section.map(
 | 
				
			||||||
 | 
					        (txt, index) =>
 | 
				
			||||||
 | 
					            new SectionCell({
 | 
				
			||||||
 | 
					                idSectionCell: index + 1,
 | 
				
			||||||
 | 
					                sectionId: title,
 | 
				
			||||||
 | 
					                contentSectionCell: txt,
 | 
				
			||||||
 | 
					                modificationDate: date,
 | 
				
			||||||
 | 
					            })
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return new Promise<SectionCell[]>((resolve) => {
 | 
				
			||||||
 | 
					        setTimeout(() => resolve(result), 500);
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getCurrentFormattedDate(): string {
 | 
				
			||||||
 | 
					    const now = new Date();
 | 
				
			||||||
 | 
					    const year = now.getFullYear();
 | 
				
			||||||
 | 
					    const month = String(now.getMonth() + 1).padStart(2, "0"); // +1 car janvier = 0
 | 
				
			||||||
 | 
					    const day = String(now.getDate()).padStart(2, "0");
 | 
				
			||||||
 | 
					    const hours = String(now.getHours()).padStart(2, "0");
 | 
				
			||||||
 | 
					    const minutes = String(now.getMinutes()).padStart(2, "0");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return `${year}-${month}-${day} ${hours}:${minutes}`;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onMounted(() => {
 | 
				
			||||||
 | 
					    fetchData(
 | 
				
			||||||
 | 
					        props.projectId,
 | 
				
			||||||
 | 
					        props.title,
 | 
				
			||||||
 | 
					        getCurrentFormattedDate(),
 | 
				
			||||||
 | 
					        IS_MOCK_MODE
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					@import "@/components/canvas/style-project.css";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.cell {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-direction: column;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    justify-content: center;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    transition: all 0.3s ease;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 5px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.expanded-content {
 | 
				
			||||||
 | 
					    justify-content: flex-start !important;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.tooltip-explain {
 | 
				
			||||||
 | 
					    position: absolute;
 | 
				
			||||||
 | 
					    bottom: 101%;
 | 
				
			||||||
 | 
					    left: 50%;
 | 
				
			||||||
 | 
					    transform: translateX(-50%);
 | 
				
			||||||
 | 
					    background-color: #333;
 | 
				
			||||||
 | 
					    color: #fff;
 | 
				
			||||||
 | 
					    padding: 6px 12px;
 | 
				
			||||||
 | 
					    font-size: 13px;
 | 
				
			||||||
 | 
					    border-radius: 6px;
 | 
				
			||||||
 | 
					    white-space: nowrap;
 | 
				
			||||||
 | 
					    opacity: 0;
 | 
				
			||||||
 | 
					    pointer-events: none;
 | 
				
			||||||
 | 
					    transition: opacity 0.3s;
 | 
				
			||||||
 | 
					    z-index: 10;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.cell:not(.expanded):hover .tooltip-explain {
 | 
				
			||||||
 | 
					    opacity: 0.9;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.cell:not(.expanded):hover {
 | 
				
			||||||
 | 
					    transform: scale(1.05);
 | 
				
			||||||
 | 
					    box-shadow: 0 8px 9px rgba(0, 0, 0, 0.2);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.cell h3 {
 | 
				
			||||||
 | 
					    font-size: 15px;
 | 
				
			||||||
 | 
					    font-weight: 500;
 | 
				
			||||||
 | 
					    font-family: "Arial", sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.p {
 | 
				
			||||||
 | 
					    font-size: 10px;
 | 
				
			||||||
 | 
					    color: #666;
 | 
				
			||||||
 | 
					    font-family: "Arial", sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.expanded {
 | 
				
			||||||
 | 
					    padding-top: 10%;
 | 
				
			||||||
 | 
					    position: fixed;
 | 
				
			||||||
 | 
					    top: 0;
 | 
				
			||||||
 | 
					    left: 0;
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    height: 100%;
 | 
				
			||||||
 | 
					    background: white;
 | 
				
			||||||
 | 
					    z-index: 10;
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    justify-content: flex-start;
 | 
				
			||||||
 | 
					    box-shadow: 0 0 10px rgba(0, 0, 0, 0.2);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.description {
 | 
				
			||||||
 | 
					    font-size: 5px;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					    word-break: break-word;
 | 
				
			||||||
 | 
					    width: 90%;
 | 
				
			||||||
 | 
					    margin: 5px 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.description + .p {
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    justify-content: center;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.edit-input {
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    height: 100%;
 | 
				
			||||||
 | 
					    min-height: 100px;
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					    border: 1px solid #ccc;
 | 
				
			||||||
 | 
					    border-radius: 5px;
 | 
				
			||||||
 | 
					    margin-top: 10px;
 | 
				
			||||||
 | 
					    box-sizing: border-box;
 | 
				
			||||||
 | 
					    margin-left: 2%;
 | 
				
			||||||
 | 
					    max-height: none;
 | 
				
			||||||
 | 
					    overflow: hidden;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.button-container {
 | 
				
			||||||
 | 
					    display: block;
 | 
				
			||||||
 | 
					    margin-top: 20px;
 | 
				
			||||||
 | 
					    justify-content: center;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    gap: 10px;
 | 
				
			||||||
 | 
					    padding-right: 1%;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.section-bloc {
 | 
				
			||||||
 | 
					    background-color: #f3f3f3;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    padding: 10px 12px;
 | 
				
			||||||
 | 
					    font-family: "Arial", sans-serif;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					    word-break: break-word;
 | 
				
			||||||
 | 
					    flex-shrink: 0;
 | 
				
			||||||
 | 
					    cursor: default;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    max-width: 100%;
 | 
				
			||||||
 | 
					    width: fit-content;
 | 
				
			||||||
 | 
					    overflow-wrap: break-word;
 | 
				
			||||||
 | 
					    box-sizing: border-box;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    min-width: 120px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.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;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.description p {
 | 
				
			||||||
 | 
					    font-size: 12px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.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;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.canvas-exit-hint {
 | 
				
			||||||
 | 
					    font-size: 0.75rem;
 | 
				
			||||||
 | 
					    color: #666;
 | 
				
			||||||
 | 
					    position: fixed;
 | 
				
			||||||
 | 
					    bottom: 10px;
 | 
				
			||||||
 | 
					    left: 0;
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    z-index: 1000;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.description-wrapper {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-direction: column;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    justify-content: center;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					    overflow: hidden;
 | 
				
			||||||
 | 
					    max-height: 100%;
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    box-sizing: border-box;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    overflow-x: hidden;
 | 
				
			||||||
 | 
					    max-height: 100%;
 | 
				
			||||||
 | 
					    box-sizing: border-box;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.custom-flow {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    flex-wrap: wrap;
 | 
				
			||||||
 | 
					    gap: 10px;
 | 
				
			||||||
 | 
					    justify-content: flex-start;
 | 
				
			||||||
 | 
					    align-items: flex-start;
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    box-sizing: border-box;
 | 
				
			||||||
 | 
					    overflow-x: hidden;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.container {
 | 
				
			||||||
 | 
					    display: grid;
 | 
				
			||||||
 | 
					    grid-template-columns: repeat(auto-fill, minmax(100px, 1fr));
 | 
				
			||||||
 | 
					    grid-auto-rows: min-content;
 | 
				
			||||||
 | 
					    grid-auto-flow: dense;
 | 
				
			||||||
 | 
					    gap: 1rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.float-up {
 | 
				
			||||||
 | 
					    transform: translateY(-10px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.float-left {
 | 
				
			||||||
 | 
					    transform: translateX(-10px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.float-right {
 | 
				
			||||||
 | 
					    transform: translateX(10px);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.wiggle {
 | 
				
			||||||
 | 
					    transform: rotate(1deg);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.tilt {
 | 
				
			||||||
 | 
					    transform: rotate(-1deg);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.from-left {
 | 
				
			||||||
 | 
					    align-self: flex-start;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.from-right {
 | 
				
			||||||
 | 
					    align-self: flex-end;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.section-bloc.from-left {
 | 
				
			||||||
 | 
					    margin-right: auto;
 | 
				
			||||||
 | 
					    margin-left: 20%;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.section-bloc.from-right {
 | 
				
			||||||
 | 
					    margin-left: auto;
 | 
				
			||||||
 | 
					    margin-right: 20%;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.explain {
 | 
				
			||||||
 | 
					    font-size: 16px;
 | 
				
			||||||
 | 
					    color: #444;
 | 
				
			||||||
 | 
					    background-color: #f9f9f9;
 | 
				
			||||||
 | 
					    padding: 7px;
 | 
				
			||||||
 | 
					    border-radius: 6px;
 | 
				
			||||||
 | 
					    margin-bottom: 20px;
 | 
				
			||||||
 | 
					    margin-top: 20px;
 | 
				
			||||||
 | 
					    line-height: 1.6;
 | 
				
			||||||
 | 
					    font-family: "Segoe UI", sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.delete-button {
 | 
				
			||||||
 | 
					    position: absolute;
 | 
				
			||||||
 | 
					    top: 5px;
 | 
				
			||||||
 | 
					    right: 10px;
 | 
				
			||||||
 | 
					    background: transparent;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    color: #a00;
 | 
				
			||||||
 | 
					    font-size: 1.2rem;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    z-index: 10;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.section-bloc {
 | 
				
			||||||
 | 
					    position: relative;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										239
									
								
								front/MyINPulse-front/src/components/canvas/HeaderCanvas.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										239
									
								
								front/MyINPulse-front/src/components/canvas/HeaderCanvas.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,239 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <header class="header">
 | 
				
			||||||
 | 
					        <a
 | 
				
			||||||
 | 
					            href="https://www.bordeaux-inp.fr/fr/lincubateur-bordeaux-inpulse"
 | 
				
			||||||
 | 
					            target="_blank"
 | 
				
			||||||
 | 
					            rel="noopener"
 | 
				
			||||||
 | 
					        >
 | 
				
			||||||
 | 
					            <img
 | 
				
			||||||
 | 
					                src="../icons/logo inpulse.png"
 | 
				
			||||||
 | 
					                alt="INPulse Logo"
 | 
				
			||||||
 | 
					                class="logo"
 | 
				
			||||||
 | 
					            />
 | 
				
			||||||
 | 
					        </a>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <div class="header-actions">
 | 
				
			||||||
 | 
					            <button class="return-button" @click="store.logout">Logout</button>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <div v-if="props.isAdmin">
 | 
				
			||||||
 | 
					                <div ref="dropdownRef" class="dropdown-wrapper">
 | 
				
			||||||
 | 
					                    <button class="contact-button" @click.stop="toggleDropdown">
 | 
				
			||||||
 | 
					                        Contact
 | 
				
			||||||
 | 
					                    </button>
 | 
				
			||||||
 | 
					                    <div
 | 
				
			||||||
 | 
					                        v-if="entrepreneurEmails.length > 0"
 | 
				
			||||||
 | 
					                        class="contact-dropdown"
 | 
				
			||||||
 | 
					                        :class="{ 'dropdown-visible': isDropdownOpen }"
 | 
				
			||||||
 | 
					                    >
 | 
				
			||||||
 | 
					                        <button @click="contactAll">Contacter tous</button>
 | 
				
			||||||
 | 
					                        <button
 | 
				
			||||||
 | 
					                            v-for="(email, index) in entrepreneurEmails"
 | 
				
			||||||
 | 
					                            :key="index"
 | 
				
			||||||
 | 
					                            @click="contactSingle(email)"
 | 
				
			||||||
 | 
					                        >
 | 
				
			||||||
 | 
					                            {{ email }}
 | 
				
			||||||
 | 
					                        </button>
 | 
				
			||||||
 | 
					                    </div>
 | 
				
			||||||
 | 
					                </div>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					    </header>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { ref, onMounted, onBeforeUnmount } from "vue";
 | 
				
			||||||
 | 
					import { store } from "@/main.ts";
 | 
				
			||||||
 | 
					import { getProjectEntrepreneurs } from "@/services/Apis/Shared.ts";
 | 
				
			||||||
 | 
					import UserEntrepreneur from "@/ApiClasses/UserEntrepreneur.ts";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const dropdownRef = ref<HTMLElement | undefined>(undefined);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const props = defineProps<{
 | 
				
			||||||
 | 
					    projectId: number;
 | 
				
			||||||
 | 
					    isAdmin: boolean;
 | 
				
			||||||
 | 
					}>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const isDropdownOpen = ref(false);
 | 
				
			||||||
 | 
					const entrepreneurEmails = ref<string[]>([]);
 | 
				
			||||||
 | 
					const entrepreneurs = ref<UserEntrepreneur[]>([]);
 | 
				
			||||||
 | 
					const IS_MOCK_MODE = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const toggleDropdown = () => {
 | 
				
			||||||
 | 
					    isDropdownOpen.value = !isDropdownOpen.value;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const fetchMockEntrepreneurs = () => {
 | 
				
			||||||
 | 
					    const mockData = [
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            userName: "Doe",
 | 
				
			||||||
 | 
					            userSurname: "John",
 | 
				
			||||||
 | 
					            primaryMail: "john.doe@example.com",
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            userName: "Smith",
 | 
				
			||||||
 | 
					            userSurname: "Anna",
 | 
				
			||||||
 | 
					            primaryMail: "anna.smith@example.com",
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            userName: "Mock",
 | 
				
			||||||
 | 
					            userSurname: "User",
 | 
				
			||||||
 | 
					            primaryMail: undefined,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    entrepreneurs.value = mockData.map((item) => new UserEntrepreneur(item));
 | 
				
			||||||
 | 
					    entrepreneurEmails.value = entrepreneurs.value
 | 
				
			||||||
 | 
					        .map((e) => e.primaryMail)
 | 
				
			||||||
 | 
					        .filter((mail): mail is string => !!mail);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    console.log("Mock entrepreneurs chargés :", entrepreneurs.value);
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const fetchEntrepreneurs = (projectId: number, useMock = false) => {
 | 
				
			||||||
 | 
					    if (useMock) {
 | 
				
			||||||
 | 
					        fetchMockEntrepreneurs();
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        if (projectId == -1) {
 | 
				
			||||||
 | 
					            console.error("No contact to show because no project was found.");
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        getProjectEntrepreneurs(
 | 
				
			||||||
 | 
					            projectId,
 | 
				
			||||||
 | 
					            (response) => {
 | 
				
			||||||
 | 
					                const rawData = response.data as Partial<UserEntrepreneur>[];
 | 
				
			||||||
 | 
					                entrepreneurs.value = rawData.map(
 | 
				
			||||||
 | 
					                    (item) => new UserEntrepreneur(item)
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					                entrepreneurEmails.value = entrepreneurs.value
 | 
				
			||||||
 | 
					                    .map((e) => e.primaryMail)
 | 
				
			||||||
 | 
					                    .filter((mail): mail is string => !!mail);
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            (error) => {
 | 
				
			||||||
 | 
					                console.error(
 | 
				
			||||||
 | 
					                    "Erreur lors de la récupération des entrepreneurs :",
 | 
				
			||||||
 | 
					                    error
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const contactAll = () => {
 | 
				
			||||||
 | 
					    const allEmails = entrepreneurEmails.value.join(", ");
 | 
				
			||||||
 | 
					    navigator.clipboard
 | 
				
			||||||
 | 
					        .writeText(allEmails)
 | 
				
			||||||
 | 
					        .then(() => {
 | 
				
			||||||
 | 
					            alert("Tous les emails copiés dans le presse-papiers !");
 | 
				
			||||||
 | 
					            window.open("https://partage.bordeaux-inp.fr/", "_blank");
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((err) => console.error("Erreur lors de la copie :", err));
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const contactSingle = (email: string) => {
 | 
				
			||||||
 | 
					    navigator.clipboard
 | 
				
			||||||
 | 
					        .writeText(email)
 | 
				
			||||||
 | 
					        .then(() => {
 | 
				
			||||||
 | 
					            alert(`Adresse copiée : ${email}`);
 | 
				
			||||||
 | 
					            window.open("https://partage.bordeaux-inp.fr/", "_blank");
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((err) => console.error("Erreur lors de la copie :", err));
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const handleClickOutside = (event: MouseEvent) => {
 | 
				
			||||||
 | 
					    if (
 | 
				
			||||||
 | 
					        isDropdownOpen.value &&
 | 
				
			||||||
 | 
					        dropdownRef.value &&
 | 
				
			||||||
 | 
					        !dropdownRef.value.contains(event.target as Node)
 | 
				
			||||||
 | 
					    ) {
 | 
				
			||||||
 | 
					        isDropdownOpen.value = false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onMounted(() => {
 | 
				
			||||||
 | 
					    if (props.isAdmin) {
 | 
				
			||||||
 | 
					        fetchEntrepreneurs(props.projectId, IS_MOCK_MODE);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    document.addEventListener("click", handleClickOutside);
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					onBeforeUnmount(() => {
 | 
				
			||||||
 | 
					    document.removeEventListener("click", handleClickOutside);
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					@import "@/components/canvas/style-project.css";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.header {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: space-between;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    padding: 10px 20px;
 | 
				
			||||||
 | 
					    background-color: #f9f9f9;
 | 
				
			||||||
 | 
					    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.logo {
 | 
				
			||||||
 | 
					    height: 50px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.header-actions {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    gap: 20px;
 | 
				
			||||||
 | 
					    position: relative;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-button,
 | 
				
			||||||
 | 
					.return-button {
 | 
				
			||||||
 | 
					    background-color: #009cde;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    padding: 10px 15px;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    font-size: 14px;
 | 
				
			||||||
 | 
					    border-radius: 5px;
 | 
				
			||||||
 | 
					    text-decoration: none;
 | 
				
			||||||
 | 
					    transition: background-color 0.2s ease;
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.return-button:hover,
 | 
				
			||||||
 | 
					.contact-button:hover {
 | 
				
			||||||
 | 
					    background-color: #007bad;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown {
 | 
				
			||||||
 | 
					    position: absolute;
 | 
				
			||||||
 | 
					    top: 100%;
 | 
				
			||||||
 | 
					    left: 0;
 | 
				
			||||||
 | 
					    background-color: #000;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					    box-shadow: 0px 4px 8px rgba(255, 255, 255, 0.2);
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					    margin-top: 5px;
 | 
				
			||||||
 | 
					    z-index: 1000;
 | 
				
			||||||
 | 
					    min-width: 200px;
 | 
				
			||||||
 | 
					    display: none;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown button {
 | 
				
			||||||
 | 
					    display: block;
 | 
				
			||||||
 | 
					    width: 100%;
 | 
				
			||||||
 | 
					    padding: 5px;
 | 
				
			||||||
 | 
					    text-align: left;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    background: none;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown button:hover {
 | 
				
			||||||
 | 
					    background-color: #009cde;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown.dropdown-visible {
 | 
				
			||||||
 | 
					    display: block;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										174
									
								
								front/MyINPulse-front/src/components/canvas/LeanCanvas.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								front/MyINPulse-front/src/components/canvas/LeanCanvas.vue
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,174 @@
 | 
				
			|||||||
 | 
					<template>
 | 
				
			||||||
 | 
					    <div class="canvas container-fluid">
 | 
				
			||||||
 | 
					        <CanvasItem
 | 
				
			||||||
 | 
					            v-for="(item, index) in items"
 | 
				
			||||||
 | 
					            :key="index"
 | 
				
			||||||
 | 
					            :title="item.title"
 | 
				
			||||||
 | 
					            :title-text="item.title_text"
 | 
				
			||||||
 | 
					            :description="item.description"
 | 
				
			||||||
 | 
					            :project-id="props.projectId"
 | 
				
			||||||
 | 
					            :class="['canvas-item', item.class, 'card', 'shadow', 'p-3']"
 | 
				
			||||||
 | 
					            :is-admin="props.isAdmin"
 | 
				
			||||||
 | 
					        />
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<script setup lang="ts">
 | 
				
			||||||
 | 
					import { ref } from "vue";
 | 
				
			||||||
 | 
					import CanvasItem from "@/components/canvas/CanvasItem.vue";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const props = defineProps<{
 | 
				
			||||||
 | 
					    projectId: number;
 | 
				
			||||||
 | 
					    isAdmin: boolean;
 | 
				
			||||||
 | 
					}>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const items = ref([
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 1,
 | 
				
			||||||
 | 
					        title_text: "1. Problème",
 | 
				
			||||||
 | 
					        description: "3 problèmes essentiels à résoudre pour le client",
 | 
				
			||||||
 | 
					        class: "Probleme",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 2,
 | 
				
			||||||
 | 
					        title_text: "2. Segments",
 | 
				
			||||||
 | 
					        description: "Les segments de clientèle visés",
 | 
				
			||||||
 | 
					        class: "Segments",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 3,
 | 
				
			||||||
 | 
					        title_text: "3. Valeur",
 | 
				
			||||||
 | 
					        description: "La proposition de valeur",
 | 
				
			||||||
 | 
					        class: "Valeur",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 4,
 | 
				
			||||||
 | 
					        title_text: "4. Solution",
 | 
				
			||||||
 | 
					        description: "Les solutions proposées",
 | 
				
			||||||
 | 
					        class: "Solution",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 5,
 | 
				
			||||||
 | 
					        title_text: "5. Avantage",
 | 
				
			||||||
 | 
					        description: "Les avantages concurrentiels",
 | 
				
			||||||
 | 
					        class: "Avantage",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 6,
 | 
				
			||||||
 | 
					        title_text: "6. Canaux",
 | 
				
			||||||
 | 
					        description: "Les canaux de distribution",
 | 
				
			||||||
 | 
					        class: "Canaux",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 7,
 | 
				
			||||||
 | 
					        title_text: "7. Indicateurs",
 | 
				
			||||||
 | 
					        description: "Les indicateurs clés de performance",
 | 
				
			||||||
 | 
					        class: "Indicateurs",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 8,
 | 
				
			||||||
 | 
					        title_text: "8. Coûts",
 | 
				
			||||||
 | 
					        description: "Les coûts associés",
 | 
				
			||||||
 | 
					        class: "Couts",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        title: 9,
 | 
				
			||||||
 | 
					        title_text: "9. Revenus",
 | 
				
			||||||
 | 
					        description: "Les sources de revenus",
 | 
				
			||||||
 | 
					        class: "Revenus",
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					]);
 | 
				
			||||||
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<style scoped>
 | 
				
			||||||
 | 
					@import "@/components/canvas/style-project.css";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.canvas {
 | 
				
			||||||
 | 
					    display: grid;
 | 
				
			||||||
 | 
					    grid-template-columns: repeat(10, minmax(0, 1fr));
 | 
				
			||||||
 | 
					    grid-auto-rows: min-content;
 | 
				
			||||||
 | 
					    gap: 12px;
 | 
				
			||||||
 | 
					    padding: 30px;
 | 
				
			||||||
 | 
					    position: relative;
 | 
				
			||||||
 | 
					    height: auto;
 | 
				
			||||||
 | 
					    max-height: none;
 | 
				
			||||||
 | 
					    box-sizing: border-box;
 | 
				
			||||||
 | 
					    overflow: visible;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@media (max-width: 768px) {
 | 
				
			||||||
 | 
					    .canvas {
 | 
				
			||||||
 | 
					        grid-template-columns: repeat(1, 1fr);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.Probleme {
 | 
				
			||||||
 | 
					    grid-column: 1 / 3;
 | 
				
			||||||
 | 
					    grid-row: 1 / 5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Segments {
 | 
				
			||||||
 | 
					    grid-column: 9 / 11;
 | 
				
			||||||
 | 
					    grid-row: 1 / 5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Valeur {
 | 
				
			||||||
 | 
					    grid-column: 5 / 7;
 | 
				
			||||||
 | 
					    grid-row: 1 / 5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Solution {
 | 
				
			||||||
 | 
					    grid-column: 3 / 5;
 | 
				
			||||||
 | 
					    grid-row: 1 / 3;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Avantage {
 | 
				
			||||||
 | 
					    grid-column: 7 / 9;
 | 
				
			||||||
 | 
					    grid-row: 1 / 3;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Canaux {
 | 
				
			||||||
 | 
					    grid-column: 7 / 9;
 | 
				
			||||||
 | 
					    grid-row: 3 / 5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Indicateurs {
 | 
				
			||||||
 | 
					    grid-column: 3 / 5;
 | 
				
			||||||
 | 
					    grid-row: 3 / 5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Couts {
 | 
				
			||||||
 | 
					    grid-column: 1 / 6;
 | 
				
			||||||
 | 
					    grid-row: 5 / 7;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Revenus {
 | 
				
			||||||
 | 
					    grid-column: 6 / 11;
 | 
				
			||||||
 | 
					    grid-row: 5 / 7;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.canvas-item {
 | 
				
			||||||
 | 
					    border: 1px solid #dee2e6;
 | 
				
			||||||
 | 
					    border-radius: 0.5rem;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.Probleme {
 | 
				
			||||||
 | 
					    background-color: #ffdddd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Segments {
 | 
				
			||||||
 | 
					    background-color: #ddffdd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Valeur {
 | 
				
			||||||
 | 
					    background-color: #ddddff;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Solution {
 | 
				
			||||||
 | 
					    background-color: #fff0b3;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Avantage {
 | 
				
			||||||
 | 
					    background-color: #d1c4e9;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Canaux {
 | 
				
			||||||
 | 
					    background-color: #b2ebf2;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Indicateurs {
 | 
				
			||||||
 | 
					    background-color: #ffe082;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Couts {
 | 
				
			||||||
 | 
					    background-color: #ffcdd2;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					.Revenus {
 | 
				
			||||||
 | 
					    background-color: #c8e6c9;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					</style>
 | 
				
			||||||
							
								
								
									
										156
									
								
								front/MyINPulse-front/src/components/canvas/style-project.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								front/MyINPulse-front/src/components/canvas/style-project.css
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,156 @@
 | 
				
			|||||||
 | 
					body {
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					    margin: 0;
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.row {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.cell {
 | 
				
			||||||
 | 
					    flex: 1;
 | 
				
			||||||
 | 
					    border: 1px solid #ddd;
 | 
				
			||||||
 | 
					    padding: 10px;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    background-color: #f1f1f1;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.produit {
 | 
				
			||||||
 | 
					    background-color: #f9e4e4;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.marche {
 | 
				
			||||||
 | 
					    background-color: #e4f1f9;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.valeur {
 | 
				
			||||||
 | 
					    background-color: #f9f4e4;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					h3 {
 | 
				
			||||||
 | 
					    margin: 0;
 | 
				
			||||||
 | 
					    font-size: 18px;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					p {
 | 
				
			||||||
 | 
					    margin: 5px 0 0;
 | 
				
			||||||
 | 
					    font-size: 14px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					body {
 | 
				
			||||||
 | 
					    font-family: Arial, sans-serif;
 | 
				
			||||||
 | 
					    margin: 0;
 | 
				
			||||||
 | 
					    padding: 0;
 | 
				
			||||||
 | 
					    background-color: #f9f9f9;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					h1 img {
 | 
				
			||||||
 | 
					    height: 80px;
 | 
				
			||||||
 | 
					    margin: 40px;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.row {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    margin-bottom: 10px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ade {
 | 
				
			||||||
 | 
					    max-width: 1200px;
 | 
				
			||||||
 | 
					    margin: 20px auto;
 | 
				
			||||||
 | 
					    padding: 20px;
 | 
				
			||||||
 | 
					    text-align: center;
 | 
				
			||||||
 | 
					    background-color: #e8f5e9;
 | 
				
			||||||
 | 
					    border: 2px solid #4caf50;
 | 
				
			||||||
 | 
					    border-radius: 10px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ade h3 {
 | 
				
			||||||
 | 
					    color: #2e7d32;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ade p {
 | 
				
			||||||
 | 
					    margin: 10px 0;
 | 
				
			||||||
 | 
					    font-size: 16px;
 | 
				
			||||||
 | 
					    color: #333;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					header {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    justify-content: space-between;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    padding: 10px 20px;
 | 
				
			||||||
 | 
					    background-color: #fff;
 | 
				
			||||||
 | 
					    border-bottom: 2px solid #ddd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					header img {
 | 
				
			||||||
 | 
					    height: 60px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					header .contact-menu {
 | 
				
			||||||
 | 
					    position: relative;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-button,
 | 
				
			||||||
 | 
					.return {
 | 
				
			||||||
 | 
					    padding: 10px 15px;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 4px;
 | 
				
			||||||
 | 
					    background-color: #2196f3;
 | 
				
			||||||
 | 
					    color: #fff;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-button:hover,
 | 
				
			||||||
 | 
					.return:hover {
 | 
				
			||||||
 | 
					    background-color: #1976d2;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Dropdown styling */
 | 
				
			||||||
 | 
					.contact-dropdown {
 | 
				
			||||||
 | 
					    position: absolute;
 | 
				
			||||||
 | 
					    right: 0;
 | 
				
			||||||
 | 
					    top: 50px;
 | 
				
			||||||
 | 
					    display: none;
 | 
				
			||||||
 | 
					    flex-direction: column;
 | 
				
			||||||
 | 
					    gap: 10px;
 | 
				
			||||||
 | 
					    padding: 15px;
 | 
				
			||||||
 | 
					    background-color: #fff;
 | 
				
			||||||
 | 
					    border: 1px solid #ddd;
 | 
				
			||||||
 | 
					    border-radius: 8px;
 | 
				
			||||||
 | 
					    box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
 | 
				
			||||||
 | 
					    z-index: 10;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown button {
 | 
				
			||||||
 | 
					    padding: 8px 12px;
 | 
				
			||||||
 | 
					    border: none;
 | 
				
			||||||
 | 
					    border-radius: 4px;
 | 
				
			||||||
 | 
					    background-color: #4caf50;
 | 
				
			||||||
 | 
					    color: #fff;
 | 
				
			||||||
 | 
					    cursor: pointer;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.contact-dropdown button:hover {
 | 
				
			||||||
 | 
					    background-color: #388e3c;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.return {
 | 
				
			||||||
 | 
					    background-color: #f44336;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.return:hover {
 | 
				
			||||||
 | 
					    background-color: #d32f2f;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.header-buttons {
 | 
				
			||||||
 | 
					    display: flex;
 | 
				
			||||||
 | 
					    align-items: center;
 | 
				
			||||||
 | 
					    gap: 15px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					a {
 | 
				
			||||||
 | 
					    color: white;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										0
									
								
								front/MyINPulse-front/src/components/contact.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								front/MyINPulse-front/src/components/contact.ts
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 5.1 KiB After Width: | Height: | Size: 39 KiB  | 
@@ -28,4 +28,46 @@ keycloakService.CallInit(() => {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// this shit made by me so i can run the canva vue app
 | 
				
			||||||
 | 
					//createApp(App).use(router).mount('#app');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TODO: fix the comment
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					function tokenInterceptor () {
 | 
				
			||||||
 | 
					    axios.interceptors.request.use(config => {
 | 
				
			||||||
 | 
					        const keycloak = useKeycloak()
 | 
				
			||||||
 | 
					        if (keycloak.authenticated) {
 | 
				
			||||||
 | 
					            // Note that this is a simple example.
 | 
				
			||||||
 | 
					            // you should be careful not to leak tokens to third parties.
 | 
				
			||||||
 | 
					            // in this example the token is added to all usage of axios.
 | 
				
			||||||
 | 
					            config.headers.Authorization = `Bearer ${keycloak.token}`
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return config
 | 
				
			||||||
 | 
					    }, error => {
 | 
				
			||||||
 | 
					        console.error("tokenInterceptor: Rejected")
 | 
				
			||||||
 | 
					        return Promise.reject(error)
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					app.use(VueKeyCloak,{
 | 
				
			||||||
 | 
					    onReady: (keycloak) => {
 | 
				
			||||||
 | 
					        console.log("Ready !")
 | 
				
			||||||
 | 
					        tokenInterceptor()
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    init: {
 | 
				
			||||||
 | 
					        onLoad: 'login-required',
 | 
				
			||||||
 | 
					        checkLoginIframe: false,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    config: {
 | 
				
			||||||
 | 
					        realm: 'test',
 | 
				
			||||||
 | 
					        url: 'http://localhost:7080',
 | 
				
			||||||
 | 
					        clientId: 'myinpulse'
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}  );
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export { store };
 | 
					export { store };
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										16
									
								
								front/MyINPulse-front/src/plugins/authStore.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								front/MyINPulse-front/src/plugins/authStore.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,16 @@
 | 
				
			|||||||
 | 
					// file: src/plugins/authStore.js
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import { useAuthStore } from "@/stores/authStore.ts";
 | 
				
			||||||
 | 
					import keycloakService from "@/services/keycloak";
 | 
				
			||||||
 | 
					import type { Pinia } from "pinia";
 | 
				
			||||||
 | 
					import type { App } from "vue";
 | 
				
			||||||
 | 
					// Setup auth store as a plugin so it can be accessed globally in our FE
 | 
				
			||||||
 | 
					const authStorePlugin = {
 | 
				
			||||||
 | 
					    install(app: App, option: { pinia: Pinia }) {
 | 
				
			||||||
 | 
					        const store = useAuthStore(option.pinia);
 | 
				
			||||||
 | 
					        app.config.globalProperties.$store = store;
 | 
				
			||||||
 | 
					        keycloakService.CallInitStore(store);
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default authStorePlugin;
 | 
				
			||||||
							
								
								
									
										0
									
								
								front/MyINPulse-front/src/router/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								front/MyINPulse-front/src/router/index.ts
									
									
									
									
									
										Normal file
									
								
							@@ -11,6 +11,47 @@ const router = createRouter({
 | 
				
			|||||||
            // 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/testComponent.vue"),
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/",
 | 
				
			||||||
 | 
					            name: "login",
 | 
				
			||||||
 | 
					            component: () => import("../components/LoginComponent.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/admin",
 | 
				
			||||||
 | 
					            name: "Admin-main",
 | 
				
			||||||
 | 
					            component: () => import("../views/AdminMain.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // route pour les canvas (made by adnane), in fact the two vue apps are separated for now
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/canvas",
 | 
				
			||||||
 | 
					            name: "canvas",
 | 
				
			||||||
 | 
					            component: () => import("../views/CanvasView.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/signup",
 | 
				
			||||||
 | 
					            name: "signup",
 | 
				
			||||||
 | 
					            component: () => import("../views/EntrepSignUp.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/JorCproject",
 | 
				
			||||||
 | 
					            name: "JorCproject",
 | 
				
			||||||
 | 
					            component: () => import("../views/JoinOrCreatProjectForEntrep.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/finalize",
 | 
				
			||||||
 | 
					            name: "finalize",
 | 
				
			||||||
 | 
					            component: () => import("../views/FinalizeAccount.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            path: "/pending-approval",
 | 
				
			||||||
 | 
					            name: "PendingApproval",
 | 
				
			||||||
 | 
					            component: () => import("@/views/PendingApproval.vue"),
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										340
									
								
								front/MyINPulse-front/src/services/Apis/Admin.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										340
									
								
								front/MyINPulse-front/src/services/Apis/Admin.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,340 @@
 | 
				
			|||||||
 | 
					import { type AxiosError, type AxiosResponse } from "axios";
 | 
				
			||||||
 | 
					import Report from "@/ApiClasses/Repport";
 | 
				
			||||||
 | 
					import ProjectDecision from "@/ApiClasses/ProjectDecision";
 | 
				
			||||||
 | 
					//import UserAdmin from "@/ApiClasses/UserAdmin";
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					    axiosInstance,
 | 
				
			||||||
 | 
					    defaultApiErrorHandler,
 | 
				
			||||||
 | 
					    defaultApiSuccessHandler,
 | 
				
			||||||
 | 
					} from "@/services/api";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Admin API
 | 
				
			||||||
 | 
					function getPendingAccounts(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/admin/pending-accounts")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function validateUserAccount(
 | 
				
			||||||
 | 
					    userId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post(`/admin/accounts/validate/${userId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getPendingProjectJoinRequests( // Not yet implemented
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/admin/request-join")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					function decideProjectJoinRequest( // Not yet implemented
 | 
				
			||||||
 | 
					    joinRequestId: number,
 | 
				
			||||||
 | 
					    decision: { isAccepted: boolean },
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post(`/admin/request-join/decision/${joinRequestId}`, decision)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getAdminProjects(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/admin/projects")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type ProjectCreatePayload = {
 | 
				
			||||||
 | 
					    projectName: string;
 | 
				
			||||||
 | 
					    logo?: string;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function addProjectManually(
 | 
				
			||||||
 | 
					    payload: ProjectCreatePayload,
 | 
				
			||||||
 | 
					    onSuccess?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onError?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post("/admin/projects", payload)
 | 
				
			||||||
 | 
					        .then((response) => onSuccess?.(response))
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => onError?.(error));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getPendingProjects(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/admin/projects/pending")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function decidePendingProject(
 | 
				
			||||||
 | 
					    decision: ProjectDecision,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post(`/admin/projects/pending/decision`, decision.toObject())
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function createAppointmentReport(
 | 
				
			||||||
 | 
					    appointmentId: number,
 | 
				
			||||||
 | 
					    reportContent: Report, // Replace 'any' with a proper type for report content if available
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post(`/admin/appointments/report/${appointmentId}`, reportContent)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function updateAppointmentReport(
 | 
				
			||||||
 | 
					    appointmentId: number,
 | 
				
			||||||
 | 
					    reportContent: Report, // Replace 'any' with a proper type for report content if available
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .put(`/admin/appointments/report/${appointmentId}`, reportContent)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getUpcomingAppointments(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/admin/appointments/upcoming")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function removeProject(
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .delete(`/admin/projects/${projectId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function grantAdminRights(
 | 
				
			||||||
 | 
					    userId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post(`/admin/make-admin/${userId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function createAdmin(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post("/admin/create-account")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export {
 | 
				
			||||||
 | 
					    axiosInstance,
 | 
				
			||||||
 | 
					    //requestJoinProject, // Not yet implemented [cite: 4]
 | 
				
			||||||
 | 
					    getPendingAccounts,
 | 
				
			||||||
 | 
					    validateUserAccount,
 | 
				
			||||||
 | 
					    getPendingProjectJoinRequests, // Not yet implemented [cite: 3]
 | 
				
			||||||
 | 
					    decideProjectJoinRequest, // Not yet implemented [cite: 3]
 | 
				
			||||||
 | 
					    getAdminProjects,
 | 
				
			||||||
 | 
					    addProjectManually,
 | 
				
			||||||
 | 
					    getPendingProjects,
 | 
				
			||||||
 | 
					    decidePendingProject,
 | 
				
			||||||
 | 
					    createAppointmentReport,
 | 
				
			||||||
 | 
					    updateAppointmentReport,
 | 
				
			||||||
 | 
					    getUpcomingAppointments,
 | 
				
			||||||
 | 
					    removeProject,
 | 
				
			||||||
 | 
					    grantAdminRights,
 | 
				
			||||||
 | 
					    createAdmin,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
							
								
								
									
										180
									
								
								front/MyINPulse-front/src/services/Apis/Entrepreneurs.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										180
									
								
								front/MyINPulse-front/src/services/Apis/Entrepreneurs.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,180 @@
 | 
				
			|||||||
 | 
					import { type AxiosError, type AxiosResponse } from "axios";
 | 
				
			||||||
 | 
					import Project from "@/ApiClasses/Project";
 | 
				
			||||||
 | 
					import SectionCell from "@/ApiClasses/SectionCell";
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					    axiosInstance,
 | 
				
			||||||
 | 
					    defaultApiErrorHandler,
 | 
				
			||||||
 | 
					    defaultApiSuccessHandler,
 | 
				
			||||||
 | 
					} from "@/services/api";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Entrepreneurs API
 | 
				
			||||||
 | 
					function getEntrepreneurProjectId(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse<number[]>) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/entrepreneur/projects")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function requestProjectCreation(
 | 
				
			||||||
 | 
					    projectDetails: Project,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post("/entrepreneur/projects/request", projectDetails.toObject())
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function addSectionCell(
 | 
				
			||||||
 | 
					    sectionCellDetails: SectionCell,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post("/entrepreneur/sectionCells", sectionCellDetails.toObject())
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function modifySectionCell(
 | 
				
			||||||
 | 
					    sectionCellId: number,
 | 
				
			||||||
 | 
					    sectionCellDetails: SectionCell,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .put(`/entrepreneur/sectionCells/${sectionCellId}`, sectionCellDetails)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function removeSectionCell(
 | 
				
			||||||
 | 
					    sectionCellId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .delete(`/entrepreneur/sectionCells/${sectionCellId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Checks if the entrepreneur has a pending project request
 | 
				
			||||||
 | 
					function checkPendingProjectRequest(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse<boolean>) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/entrepreneur/projects/has-pending-request")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Checks if the entrepreneur has an active project
 | 
				
			||||||
 | 
					function checkIfProjectIsActive(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse<boolean>) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/entrepreneur/projects/project-is-active")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export {
 | 
				
			||||||
 | 
					    getEntrepreneurProjectId,
 | 
				
			||||||
 | 
					    requestProjectCreation,
 | 
				
			||||||
 | 
					    addSectionCell,
 | 
				
			||||||
 | 
					    modifySectionCell,
 | 
				
			||||||
 | 
					    removeSectionCell,
 | 
				
			||||||
 | 
					    checkPendingProjectRequest,
 | 
				
			||||||
 | 
					    checkIfProjectIsActive,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
							
								
								
									
										157
									
								
								front/MyINPulse-front/src/services/Apis/Shared.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								front/MyINPulse-front/src/services/Apis/Shared.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,157 @@
 | 
				
			|||||||
 | 
					import { type AxiosError, type AxiosResponse } from "axios";
 | 
				
			||||||
 | 
					import Appointment from "@/ApiClasses/Appointment";
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					    axiosInstance,
 | 
				
			||||||
 | 
					    defaultApiErrorHandler,
 | 
				
			||||||
 | 
					    defaultApiSuccessHandler,
 | 
				
			||||||
 | 
					} from "@/services/api";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Shared API
 | 
				
			||||||
 | 
					function getSectionCellsByDate(
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    sectionId: number,
 | 
				
			||||||
 | 
					    date: string, // Use string for date in 'YYYY-MM-DD HH:mm' format
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get(`/shared/projects/sectionCells/${projectId}/${sectionId}/${date}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getProjectEntrepreneurs(
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get(`/shared/projects/entrepreneurs/${projectId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getProjectAdmin(
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get(`/shared/projects/admin/${projectId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getProjectAppointments(
 | 
				
			||||||
 | 
					    projectId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get(`/shared/projects/appointments/${projectId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getAppointmentReport(
 | 
				
			||||||
 | 
					    appointmentId: number,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get(`/shared/appointments/report/${appointmentId}`)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function requestAppointment(
 | 
				
			||||||
 | 
					    appointmentDetails: Appointment, // Replace 'any' with a proper type for appointment details if available
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post("/shared/appointments/request", appointmentDetails)
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export {
 | 
				
			||||||
 | 
					    getSectionCellsByDate,
 | 
				
			||||||
 | 
					    getProjectEntrepreneurs,
 | 
				
			||||||
 | 
					    getProjectAdmin,
 | 
				
			||||||
 | 
					    getProjectAppointments,
 | 
				
			||||||
 | 
					    getAppointmentReport,
 | 
				
			||||||
 | 
					    requestAppointment,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
							
								
								
									
										103
									
								
								front/MyINPulse-front/src/services/Apis/Unauth.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								front/MyINPulse-front/src/services/Apis/Unauth.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,103 @@
 | 
				
			|||||||
 | 
					import { type AxiosError, type AxiosResponse } from "axios";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					    axiosInstance,
 | 
				
			||||||
 | 
					    defaultApiErrorHandler,
 | 
				
			||||||
 | 
					    defaultApiSuccessHandler,
 | 
				
			||||||
 | 
					} from "@/services/api";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Unauth API
 | 
				
			||||||
 | 
					function finalizeAccount(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post("/unauth/finalize")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// function requestJoinProject( // Not yet implemented [cite: 4]
 | 
				
			||||||
 | 
					//     projectId: number,
 | 
				
			||||||
 | 
					//     onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					//     onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					// ): void {
 | 
				
			||||||
 | 
					//     axiosInstance
 | 
				
			||||||
 | 
					//         .post(`/unauth/request-join/${projectId}`)
 | 
				
			||||||
 | 
					//          .then((response) => {
 | 
				
			||||||
 | 
					//             if (onSuccessHandler) {
 | 
				
			||||||
 | 
					//                 onSuccessHandler(response);
 | 
				
			||||||
 | 
					//             } else {
 | 
				
			||||||
 | 
					//                 defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					//             }
 | 
				
			||||||
 | 
					//         })
 | 
				
			||||||
 | 
					//         .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					//             if (onErrorHandler) {
 | 
				
			||||||
 | 
					//                 onErrorHandler(error);
 | 
				
			||||||
 | 
					//             } else {
 | 
				
			||||||
 | 
					//                 defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					//             }
 | 
				
			||||||
 | 
					//         });
 | 
				
			||||||
 | 
					// }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function getAllEntrepreneurs(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/unauth/getAllEntrepreneurs")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function checkPending(
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse<boolean>) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .get("/unauth/check-if-not-pending")
 | 
				
			||||||
 | 
					        .then((response) => {
 | 
				
			||||||
 | 
					            if (onSuccessHandler) {
 | 
				
			||||||
 | 
					                onSuccessHandler(response);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiSuccessHandler(response);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					        .catch((error: AxiosError) => {
 | 
				
			||||||
 | 
					            if (onErrorHandler) {
 | 
				
			||||||
 | 
					                onErrorHandler(error);
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                defaultApiErrorHandler(error);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					export {
 | 
				
			||||||
 | 
					    finalizeAccount,
 | 
				
			||||||
 | 
					    getAllEntrepreneurs,
 | 
				
			||||||
 | 
					    checkPending,
 | 
				
			||||||
 | 
					    // requestJoinProject, // Not yet implemented [cite: 4]
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
@@ -1,6 +1,7 @@
 | 
				
			|||||||
import axios, { type AxiosError, type AxiosResponse } from "axios";
 | 
					import axios, { type AxiosError, type AxiosResponse } 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";
 | 
				
			||||||
 | 
					import router from "@/router/router";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const axiosInstance = axios.create({
 | 
					const axiosInstance = axios.create({
 | 
				
			||||||
    baseURL: import.meta.env.VITE_BACKEND_URL,
 | 
					    baseURL: import.meta.env.VITE_BACKEND_URL,
 | 
				
			||||||
@@ -9,6 +10,19 @@ const axiosInstance = axios.create({
 | 
				
			|||||||
    },
 | 
					    },
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					axiosInstance.interceptors.request.use(
 | 
				
			||||||
 | 
					    (config) => {
 | 
				
			||||||
 | 
					        const token = store.user?.token; // Récupérez le token depuis le store
 | 
				
			||||||
 | 
					        if (token) {
 | 
				
			||||||
 | 
					            config.headers["Authorization"] = `Bearer ${token}`; // Ajoutez le token dans l'en-tête
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return config;
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    (error) => {
 | 
				
			||||||
 | 
					        return Promise.reject(error);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
axiosInstance.interceptors.response.use(
 | 
					axiosInstance.interceptors.response.use(
 | 
				
			||||||
    (response) => response, // Directly return successful responses.
 | 
					    (response) => response, // Directly return successful responses.
 | 
				
			||||||
    async (error) => {
 | 
					    async (error) => {
 | 
				
			||||||
@@ -19,19 +33,17 @@ axiosInstance.interceptors.response.use(
 | 
				
			|||||||
            !originalRequest._retry &&
 | 
					            !originalRequest._retry &&
 | 
				
			||||||
            store.authenticated
 | 
					            store.authenticated
 | 
				
			||||||
        ) {
 | 
					        ) {
 | 
				
			||||||
            originalRequest._retry = true; // Mark the request as retried to avoid infinite loops.
 | 
					            originalRequest._retry = true;
 | 
				
			||||||
            try {
 | 
					            try {
 | 
				
			||||||
                await store.refreshUserToken();
 | 
					                await store.refreshUserToken();
 | 
				
			||||||
                // 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);
 | 
				
			||||||
            } catch (refreshError) {
 | 
					            } catch (refreshError) {
 | 
				
			||||||
                // 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";
 | 
					                router.push("/login");
 | 
				
			||||||
                return Promise.reject(refreshError);
 | 
					                return Promise.reject(refreshError);
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@@ -41,7 +53,9 @@ axiosInstance.interceptors.response.use(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// TODO: spawn a error modal
 | 
					// TODO: spawn a error modal
 | 
				
			||||||
function defaultApiErrorHandler(err: AxiosError) {
 | 
					function defaultApiErrorHandler(err: AxiosError) {
 | 
				
			||||||
    addNewMessage(err.message, color.Red);
 | 
					    const errorMessage =
 | 
				
			||||||
 | 
					        (err.response?.data as { message?: string })?.message ?? err.message;
 | 
				
			||||||
 | 
					    addNewMessage(errorMessage, color.Red);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function defaultApiSuccessHandler(response: AxiosResponse) {
 | 
					function defaultApiSuccessHandler(response: AxiosResponse) {
 | 
				
			||||||
@@ -65,4 +79,36 @@ function callApi(
 | 
				
			|||||||
        );
 | 
					        );
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export { callApi };
 | 
					function postApi(
 | 
				
			||||||
 | 
					    endpoint: string,
 | 
				
			||||||
 | 
					    data: unknown, //to fix eslint issue, go back here if errors occurs later
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .post(endpoint, data)
 | 
				
			||||||
 | 
					        .then(onSuccessHandler ?? defaultApiSuccessHandler)
 | 
				
			||||||
 | 
					        .catch(onErrorHandler ?? defaultApiErrorHandler);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function deleteApi(
 | 
				
			||||||
 | 
					    endpoint: string,
 | 
				
			||||||
 | 
					    onSuccessHandler?: (response: AxiosResponse) => void,
 | 
				
			||||||
 | 
					    onErrorHandler?: (error: AxiosError) => void
 | 
				
			||||||
 | 
					): void {
 | 
				
			||||||
 | 
					    axiosInstance
 | 
				
			||||||
 | 
					        .delete(endpoint)
 | 
				
			||||||
 | 
					        .then(onSuccessHandler ?? defaultApiSuccessHandler)
 | 
				
			||||||
 | 
					        .catch(onErrorHandler ?? defaultApiErrorHandler);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//export { axiosInstance, callApi, postApi, deleteApi };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export {
 | 
				
			||||||
 | 
					    axiosInstance,
 | 
				
			||||||
 | 
					    defaultApiErrorHandler,
 | 
				
			||||||
 | 
					    defaultApiSuccessHandler,
 | 
				
			||||||
 | 
					    callApi,
 | 
				
			||||||
 | 
					    postApi,
 | 
				
			||||||
 | 
					    deleteApi,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										24
									
								
								front/MyINPulse-front/src/services/tools.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								front/MyINPulse-front/src/services/tools.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
				
			|||||||
 | 
					import { jwtDecode } from "jwt-decode";
 | 
				
			||||||
 | 
					import { store } from "@/main";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type TokenPayload = {
 | 
				
			||||||
 | 
					    realm_access?: {
 | 
				
			||||||
 | 
					        roles?: string[];
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function isAdmin(): boolean {
 | 
				
			||||||
 | 
					    if (store.authenticated && store.user.token) {
 | 
				
			||||||
 | 
					        const decoded = jwtDecode<TokenPayload>(store.user.token);
 | 
				
			||||||
 | 
					        const roles = decoded.realm_access?.roles || [];
 | 
				
			||||||
 | 
					        if (roles.includes("MyINPulse-admin")) {
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export { isAdmin };
 | 
				
			||||||
@@ -54,7 +54,7 @@ const useAuthStore = defineStore("storeAuth", {
 | 
				
			|||||||
        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 + "/" //redirect to login page instead of test...
 | 
				
			||||||
                );
 | 
					                );
 | 
				
			||||||
                await this.clearUserData();
 | 
					                await this.clearUserData();
 | 
				
			||||||
            } catch (error) {
 | 
					            } catch (error) {
 | 
				
			||||||
 
 | 
				
			|||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user