Compare commits
	
		
			52 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ba0ab6d57b | ||
|   | ef17ad203d | ||
|   | 19f08d36cf | ||
|   | 141c8f1659 | ||
|   | dd677afff0 | ||
|   | cc0d502229 | ||
|   | 0d32740ec7 | ||
|   | 839236ecd1 | ||
|   | 4c7a9e27e3 | ||
|   | 08879a6860 | ||
|   | f72645e05c | ||
|   | e834e0f096 | ||
|   | 4eca38ad24 | ||
|   | 7543f07562 | ||
|   | a90be58013 | ||
|   | 93a643d10d | ||
|   | 85c52fac2a | ||
|   | 8239e387fb | ||
|   | b4e17d6853 | ||
|   | ca16e3c43a | ||
|   | 5f9247391a | ||
|   | ee12030cf6 | ||
|   | b062fb0b97 | ||
|   | 01c72af17a | ||
|   | b28eb6d3b7 | ||
|   | 5d2392ad9e | ||
|   | efb334f8de | ||
|   | d4b1c6a9e1 | ||
|   | f7a62560c7 | ||
|   | e7393c0192 | ||
|   | bc647ce9be | ||
|   | af6cb7d1ec | ||
|   | a88e7d2ce5 | ||
| 3d78b90ff1 | |||
| 3d7ee42ca8 | |||
| 77acd2060b | |||
| 88f65f0806 | |||
| 6df71e3341 | |||
| 263a4ae4c0 | |||
| 57a48de52f | |||
| ae9f00a60a | |||
| 93b95becb7 | |||
| bd66703a08 | |||
| 86fec254aa | |||
| 6f16bffe31 | |||
| 1fc1b5d35d | |||
| 7d6e3bb8c8 | |||
| 11b8b575e4 | |||
| dfa245e842 | |||
| ed94562372 | |||
| 5fee3a1390 | |||
| 96bc7ec052 | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -53,6 +53,8 @@ testem.log | ||||
| # System Files | ||||
| .DS_Store | ||||
| Thumbs.db | ||||
| /env_dev/data | ||||
| /env_dev/dataMongo | ||||
|  | ||||
| backPY/env | ||||
|  | ||||
| @@ -62,3 +64,5 @@ __pycache__ | ||||
|  | ||||
| .design/ | ||||
| .vscode/ | ||||
| front/storybook-static | ||||
| back/bin | ||||
|   | ||||
| @@ -1,7 +1,32 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| version_file="../version.txt" | ||||
|  | ||||
| # update new release dependency | ||||
| cd back | ||||
| mvn versions:set -DnewVersion=$(cat ../version.txt) | ||||
| # update the Maven version number | ||||
| mvn versions:set -DnewVersion=$(sed 's/dev/SNAPSHOT/g' $version_file) | ||||
| if grep -q "DEV" "$version_file"; then | ||||
|     # update all versions release of dependency | ||||
|     mvn versions:use-latest-releases  | ||||
|     # update our manage dependency as snapshoot | ||||
|     mvn versions:use-latest-versions -Dincludes=kangaroo-and-rabbit | ||||
| else | ||||
|     # update our manage dependency as release (must be done before) | ||||
|     mvn versions:use-latest-releases -Dincludes=kangaroo-and-rabbit | ||||
| fi | ||||
| cd - | ||||
|  | ||||
|  | ||||
| cd front | ||||
| if grep -q "dev" "$version_file"; then | ||||
|     # update all dependency | ||||
|     pnpm install | ||||
|     pnpm run update_packages | ||||
| else | ||||
|     # in case of release ==> can not do it automatically ... | ||||
|     echo not implemented | ||||
| fi | ||||
|  | ||||
| cd - | ||||
|  | ||||
|   | ||||
							
								
								
									
										104
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										104
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,25 +1,33 @@ | ||||
| ###################################################################################### | ||||
| ## | ||||
| ## buyilding-end install applications: | ||||
| ## building-end install applications: | ||||
| ## | ||||
| ###################################################################################### | ||||
| FROM archlinux:base-devel AS builder | ||||
| FROM archlinux:base-devel AS common | ||||
| # update system | ||||
| RUN pacman -Syu --noconfirm && pacman-db-upgrade \ | ||||
|     && pacman -S --noconfirm jdk-openjdk maven npm \ | ||||
|     && pacman -S --noconfirm jdk-openjdk wget\ | ||||
|     && pacman -Scc --noconfirm | ||||
|  | ||||
| ENV PATH /tmp/node_modules/.bin:$PATH | ||||
| WORKDIR /tmp | ||||
|  | ||||
| FROM common AS builder | ||||
| # update system | ||||
| RUN pacman -Syu --noconfirm && pacman-db-upgrade \ | ||||
|     && pacman -S --noconfirm maven npm pnpm \ | ||||
|     && pacman -Scc --noconfirm | ||||
|  | ||||
| ENV PATH=/tmp/node_modules/.bin:$PATH | ||||
|  | ||||
| ###################################################################################### | ||||
| ## | ||||
| ## Build back: | ||||
| ## | ||||
| ###################################################################################### | ||||
| FROM builder AS buildBack | ||||
| COPY back/pom.xml /tmp | ||||
| COPY back/src /tmp/src/ | ||||
| FROM builder AS build_back | ||||
| COPY back/pom.xml ./ | ||||
| COPY back/Formatter.xml ./ | ||||
| COPY back/src ./src/ | ||||
| RUN mvn clean compile assembly:single | ||||
|  | ||||
| ###################################################################################### | ||||
| @@ -27,26 +35,44 @@ RUN mvn clean compile assembly:single | ||||
| ## Build front: | ||||
| ## | ||||
| ###################################################################################### | ||||
| FROM builder AS buildFront | ||||
| FROM builder AS dependency_front | ||||
|  | ||||
| ADD front/package-lock.json \ | ||||
|     front/package.json \ | ||||
|     front/karma.conf.js \ | ||||
|     front/protractor.conf.js \ | ||||
|     /tmp/ | ||||
| RUN echo "@kangaroo-and-rabbit:registry=https://gitea.atria-soft.org/api/packages/kangaroo-and-rabbit/npm/" > /root/.npmrc | ||||
|  | ||||
| COPY front/package.json \ | ||||
|     front/pnpm-lock.yaml \ | ||||
|     ./ | ||||
| COPY front/src/theme ./src/theme | ||||
|  | ||||
| # install and cache app dependencies | ||||
| RUN npm install | ||||
| RUN pnpm install --prod=false | ||||
|  | ||||
| ADD front/e2e \ | ||||
|     front/tsconfig.json \ | ||||
|     front/tslint.json \ | ||||
|     front/angular.json \ | ||||
|     /tmp/ | ||||
| ADD front/src /tmp/src | ||||
| ############################################################### | ||||
| ## Install sources | ||||
| ############################################################### | ||||
| FROM dependency_front AS load_sources_front | ||||
|  | ||||
| # generate build | ||||
| RUN ng build --output-path=dist --configuration=production --base-href=/karideo/ --deploy-url=/karideo/  | ||||
| # JUST to get the version of the application and his sha... | ||||
| COPY \ | ||||
|   front/tsconfig.json \ | ||||
|   front/tsconfig.node.json \ | ||||
|   front/vite.config.mts \ | ||||
|   front/index.html \ | ||||
|   ./ | ||||
|  | ||||
| COPY front/public ./public | ||||
| COPY front/src ./src | ||||
|  | ||||
| # We are not in prod mode ==> we need to overwrite the production env. | ||||
| ARG env=front/.env.production | ||||
| COPY ${env} .env | ||||
|  | ||||
| ############################################################### | ||||
| ## Build the sources | ||||
| ############################################################### | ||||
| FROM load_sources_front AS build_front | ||||
| # build in bundle mode all the application | ||||
| RUN pnpm static:build | ||||
|  | ||||
| ###################################################################################### | ||||
| ## | ||||
| @@ -54,26 +80,24 @@ RUN ng build --output-path=dist --configuration=production --base-href=/karideo/ | ||||
| ## | ||||
| ###################################################################################### | ||||
|  | ||||
| FROM bellsoft/liberica-openjdk-alpine:latest | ||||
| # add wget to manage the health check... | ||||
| RUN apk add --no-cache wget | ||||
| FROM bellsoft/liberica-openjdk-alpine-musl:latest | ||||
|  | ||||
| #FROM archlinux:base | ||||
| #RUN pacman -Syu --noconfirm && pacman-db-upgrade | ||||
| ## install package | ||||
| #RUN pacman -S --noconfirm jdk-openjdk wget | ||||
| ## intall npm | ||||
| #RUN pacman -S --noconfirm npm | ||||
| ## clean all the caches Need only on the release environment | ||||
| #RUN pacman -Scc --noconfirm | ||||
| RUN apk add --no-cache wget \ | ||||
|   && addgroup -g 1000 user \ | ||||
|   && adduser --system -u 1000 -G user user | ||||
|  | ||||
| ENV LANG=C.UTF-8 | ||||
|  | ||||
| COPY --from=buildBack /tmp/out/maven/*.jar /application/application.jar | ||||
| COPY --from=buildFront /tmp/dist /application/front/ | ||||
|  | ||||
| WORKDIR /application/ | ||||
|  | ||||
| EXPOSE 80 | ||||
| WORKDIR /application/ | ||||
| RUN chown user:user -R /application | ||||
|  | ||||
| CMD ["java", "-Xms64M", "-Xmx1G", "-cp", "/application/application.jar", "org.kar.karideo.WebLauncher"] | ||||
| # To verify health-check: docker inspect --format "{{json .State.Health }}" YOUR_SERVICE_NAME | jq | ||||
| HEALTHCHECK --start-period=10s --start-interval=2s --interval=30s --timeout=5s --retries=10 \ | ||||
|    CMD wget --no-verbose --tries=1 --spider http://localhost:80/api/health_check || exit 1 | ||||
|  | ||||
| CMD ["java", "-Xms128M", "-Xmx1G", "-cp", "/application/application.jar", "org.atriasoft.karideo.WebLauncher"] | ||||
|  | ||||
| COPY --chown=user:user --from=build_back /tmp/out/maven/*.jar /application/application.jar | ||||
| COPY --chown=user:user --from=build_front /tmp/dist /application/front/ | ||||
|  | ||||
| USER user | ||||
|   | ||||
							
								
								
									
										145
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| Karideo | ||||
| ======= | ||||
|  | ||||
| **K**angaroo **A**nd **R**abbit (v)ideo is a simple framework to propose video streaming for personal network | ||||
|  | ||||
| Run in local: | ||||
| ============= | ||||
|  | ||||
| Start tools | ||||
| ----------- | ||||
|  | ||||
| Start the server basic interfaces: (DB(mySQL), Adminer) | ||||
|  | ||||
| ```{.bash} | ||||
| # start the Bdd interface (no big data > 50Mo) | ||||
| docker compose -f env_dev/docker-compose.yaml up -d | ||||
| ``` | ||||
|  | ||||
| Start the Back-end: | ||||
| ------------------- | ||||
|  | ||||
| backend is developed in JAVA | ||||
|  | ||||
| The first step is configuring your JAVA version (or select the JVM with the OS) | ||||
| ```bash | ||||
| export PATH=$(ls -d --color=never /usr/lib/jvm/java-2*-openjdk)/bin:$PATH | ||||
| ``` | ||||
|  | ||||
| Install the dependency: | ||||
| ```bash | ||||
| mvn install | ||||
| ``` | ||||
|  | ||||
| Run the test | ||||
| ```bash | ||||
| mvn test | ||||
| ``` | ||||
|  | ||||
| Install it for external use | ||||
| ```bash | ||||
| mvn install | ||||
| ``` | ||||
|  | ||||
| Execute the local server: | ||||
| ```bash | ||||
| mvn exec:java@dev-mode | ||||
| ``` | ||||
|  | ||||
| Start the Front-end: | ||||
| -------------------- | ||||
|  | ||||
| backend is developed in JAVA | ||||
| ```bash | ||||
| cd front | ||||
| pnpm install | ||||
| pnpm dev | ||||
| ``` | ||||
|  | ||||
| Display the result: | ||||
| ------------------- | ||||
|  | ||||
| [show the webpage: http://localhost:4203](http://localhost:4203) | ||||
|  | ||||
| Some other dev tools: | ||||
| ===================== | ||||
|  | ||||
| Format code: | ||||
| ------------ | ||||
|  | ||||
| ```bash | ||||
| export PATH=$(ls -d --color=never /usr/lib/jvm/java-2*-openjdk)/bin:$PATH | ||||
| mvn formatter:format | ||||
| mvn test | ||||
| ``` | ||||
|  | ||||
| Tools in production mode | ||||
| ======================== | ||||
|  | ||||
| Changing the Log Level | ||||
| ---------------------- | ||||
|  | ||||
| In a production environment, you can adjust the log level to help diagnose bugs more effectively. | ||||
|  | ||||
| The available log levels are: | ||||
| | **Log Level Tag** | **org.kar.karideo** | **org.kar.archidata** | **other** | | ||||
| | ----------------- | ------------------- | --------------------- | --------- | | ||||
| | `prod`            | INFO                | INFO                  | INFO      | | ||||
| | `prod-debug`      | DEBUG               | INFO                  | INFO      | | ||||
| | `prod-trace`      | TRACE               | DEBUG                 | INFO      | | ||||
| | `prod-trace-full` | TRACE               | TRACE                 | INFO      | | ||||
| | `dev`             | TRACE               | DEBUG                 | INFO      | | ||||
|  | ||||
|  | ||||
| Manual set in production: | ||||
| ========================= | ||||
|  | ||||
| Connect on the registry | ||||
| ------------------------ | ||||
|  | ||||
| To log-in and log-out from the registry: | ||||
| ```bash | ||||
| export REGISTRY_ADDRESS=gitea.atria-soft.org | ||||
| docker login -u <<YOUR_USER_NAME>> ${REGISTRY_ADDRESS} | ||||
| docker logout ${REGISTRY_ADDRESS} | ||||
| ``` | ||||
|  | ||||
| pull the root image of dockers | ||||
| ------------------------------ | ||||
|  | ||||
| ```bash | ||||
| docker pull archlinux:base-devel | ||||
| docker pull bellsoft/liberica-openjdk-alpine:latest | ||||
| ``` | ||||
|  | ||||
| Deploy on mongodb | ||||
| ----------------- | ||||
|  | ||||
| connect on the mongosh interface: | ||||
| ```bash | ||||
| docker compose exec kar_mongodb_service mongosh -u root | ||||
| ``` | ||||
|  | ||||
| Create the base and the user | ||||
| ``` | ||||
| use karideo | ||||
| db.createUser({ | ||||
|   user: "karideo", | ||||
|   pwd: "your_password", | ||||
|   roles: [ | ||||
|     { role: "readWrite", db: "karideo" } | ||||
|   ] | ||||
| }) | ||||
| ``` | ||||
|  | ||||
| Create the version | ||||
| ------------------ | ||||
|  | ||||
| Execute in the local folder: (use ```dev``` for development and ```latest``` for production release) | ||||
|  | ||||
| ```bash | ||||
| export TAG_DOCKER=latest | ||||
| export REGISTRY_ADDRESS=gitea.atria-soft.org | ||||
| docker build -t ${REGISTRY_ADDRESS}/kangaroo-and-rabbit/karideo:${TAG_DOCKER} . | ||||
| docker push ${REGISTRY_ADDRESS}/kangaroo-and-rabbit/karideo:${TAG_DOCKER} | ||||
| ``` | ||||
| @@ -53,6 +53,9 @@ Checkstyle configuration that checks the sun coding conventions. | ||||
|     <module name="LambdaParameterName"/> | ||||
|     <module name="Regexp"/> | ||||
|     <module name="RegexpSinglelineJava"/> | ||||
|     <module name="UnusedPrivateField"> | ||||
|         <property name="ignorePattern" value="LOGGER"/> | ||||
|     </module> | ||||
|   </module> | ||||
|   <module name="BeforeExecutionExclusionFileFilter"> | ||||
|     <property name="fileNamePattern" value="module\-info\.java$"/> | ||||
|   | ||||
| @@ -1,21 +0,0 @@ | ||||
| FROM maven:3.6.3-openjdk-16 AS build | ||||
|  | ||||
| COPY pom.xml /tmp/ | ||||
| COPY src /tmp/src/ | ||||
| WORKDIR /tmp/ | ||||
| RUN mvn clean compile assembly:single | ||||
|  | ||||
| FROM bellsoft/liberica-openjdk-alpine:latest | ||||
| ENV LANG=C.UTF-8 | ||||
|  | ||||
| # add wget to manage the health check... | ||||
| RUN apk add --no-cache wget | ||||
|  | ||||
| RUN mkdir /application/ | ||||
| COPY --from=build /tmp/out/maven/*.jar /application/application.jar | ||||
| WORKDIR /application/ | ||||
|  | ||||
| EXPOSE 18080 | ||||
|  | ||||
| CMD ["java", "-Xms64M", "-Xmx1G", "-cp", "/application/application.jar", "org.kar.karideo.WebLauncher"] | ||||
|  | ||||
| @@ -11,7 +11,7 @@ mvn package | ||||
| // download all dependency in out/maven/dependency | ||||
| mvn dependency:copy-dependencies | ||||
|  | ||||
| java -cp out/maven/kar-karideo-0.1.0.jar  org.kar.karideo.WebLauncher | ||||
| java -cp out/maven/kar-karideo-0.1.0.jar  org.atriasoft.karideo.WebLauncher | ||||
|  | ||||
|  | ||||
| // create a single package jar | ||||
| @@ -19,7 +19,7 @@ mvn clean compile assembly:single | ||||
|  | ||||
|  | ||||
|  | ||||
| java -cp out/maven/karideo-0.1.0-jar-with-dependencies.jar org.kar.karideo.WebLauncher | ||||
| java -cp out/maven/karideo-0.1.0-jar-with-dependencies.jar org.atriasoft.karideo.WebLauncher | ||||
|  | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										196
									
								
								back/pom.xml
									
									
									
									
									
								
							
							
						
						
									
										196
									
								
								back/pom.xml
									
									
									
									
									
								
							| @@ -1,31 +1,47 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
| 	<modelVersion>4.0.0</modelVersion> | ||||
| 	<groupId>org.kar</groupId> | ||||
| 	<groupId>org.atriasoft</groupId> | ||||
| 	<artifactId>karideo</artifactId> | ||||
| 	<version>0.3.0</version> | ||||
| 	<properties> | ||||
| 		<maven.compiler.version>3.1</maven.compiler.version> | ||||
| 		<maven.compiler.source>21</maven.compiler.source> | ||||
| 		<maven.compiler.target>21</maven.compiler.target> | ||||
| 		<maven.compiler.version>3.13.0</maven.compiler.version> | ||||
| 		<maven.compiler.source>23</maven.compiler.source> | ||||
| 		<maven.compiler.target>23</maven.compiler.target> | ||||
| 		<maven.dependency.version>3.1.1</maven.dependency.version> | ||||
| 	</properties> | ||||
| 	<repositories> | ||||
| 		<repository> | ||||
| 			<id>gitea</id> | ||||
| 			<url>https://gitea.atria-soft.org/api/packages/kangaroo-and-rabbit/maven</url> | ||||
| 		</repository> | ||||
| 	</repositories> | ||||
| 	<dependencies> | ||||
| 		<dependency> | ||||
| 			<groupId>kangaroo-and-rabbit</groupId> | ||||
| 			<groupId>org.atria-soft</groupId> | ||||
| 			<artifactId>archidata</artifactId> | ||||
| 			<version>0.7.3</version> | ||||
| 			<version>0.37.2</version> | ||||
| 		</dependency> | ||||
| 		<!-- Loopback of logger JDK logging API to SLF4J --> | ||||
| 		<dependency> | ||||
| 			<groupId>org.slf4j</groupId> | ||||
| 			<artifactId>slf4j-simple</artifactId> | ||||
| 			<version>2.0.9</version> | ||||
| 			<artifactId>jul-to-slf4j</artifactId> | ||||
| 			<version>2.1.0-alpha1</version> | ||||
| 		</dependency> | ||||
| 		<!-- generic logger of SLF4J to console (in color) --> | ||||
| 		<dependency> | ||||
| 			<groupId>ch.qos.logback</groupId> | ||||
| 			<artifactId>logback-classic</artifactId> | ||||
| 			<version>1.5.19</version> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>xerces</groupId> | ||||
| 			<artifactId>xercesImpl</artifactId> | ||||
| 			<version>2.12.2</version> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>org.codehaus.janino</groupId> | ||||
| 			<artifactId>janino</artifactId> | ||||
| 			<version>3.1.12</version> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>com.fasterxml.jackson.datatype</groupId> | ||||
| 			<artifactId>jackson-datatype-jsr310</artifactId> | ||||
| 			<version>2.20.0</version> | ||||
| 		</dependency> | ||||
| 		<!-- | ||||
| 		************************************************************  | ||||
| @@ -35,15 +51,25 @@ | ||||
| 		<dependency> | ||||
| 			<groupId>org.junit.jupiter</groupId> | ||||
| 			<artifactId>junit-jupiter-api</artifactId> | ||||
| 			<version>5.10.1</version> | ||||
| 			<version>6.0.0</version> | ||||
| 			<scope>test</scope> | ||||
| 		</dependency> | ||||
| 	    <dependency> | ||||
| 	        <groupId>org.junit.jupiter</groupId> | ||||
| 	        <artifactId>junit-jupiter-engine</artifactId> | ||||
| 			<version>5.10.1</version> | ||||
| 	        <scope>test</scope> | ||||
| 	    </dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>org.junit.jupiter</groupId> | ||||
| 			<artifactId>junit-jupiter-engine</artifactId> | ||||
| 			<version>6.0.0</version> | ||||
| 			<scope>test</scope> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>net.revelc.code.formatter</groupId> | ||||
| 			<artifactId>formatter-maven-plugin</artifactId> | ||||
| 			<version>2.29.0</version> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>org.apache.maven.plugins</groupId> | ||||
| 			<artifactId>maven-checkstyle-plugin</artifactId> | ||||
| 			<version>3.6.0</version> | ||||
| 		</dependency> | ||||
| 	</dependencies> | ||||
| 	<build> | ||||
| 		<sourceDirectory>src</sourceDirectory> | ||||
| @@ -72,16 +98,45 @@ | ||||
| 			<plugin> | ||||
| 				<groupId>org.codehaus.mojo</groupId> | ||||
| 				<artifactId>exec-maven-plugin</artifactId> | ||||
| 				<version>1.4.0</version> | ||||
| 				<version>3.2.0</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>prod-mode</id> | ||||
| 						<goals> | ||||
| 							<goal>java</goal> | ||||
| 						</goals> | ||||
| 						<configuration> | ||||
| 							<mainClass>org.atriasoft.karideo.WebLauncher</mainClass> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>dev-mode</id> | ||||
| 						<goals> | ||||
| 							<goal>java</goal> | ||||
| 						</goals> | ||||
| 						<configuration> | ||||
| 							<mainClass>org.atriasoft.karideo.WebLauncherLocal</mainClass> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>generate-api</id> | ||||
| 						<goals> | ||||
| 							<goal>java</goal> | ||||
| 						</goals> | ||||
| 						<configuration> | ||||
| 							<mainClass>org.atriasoft.karideo.GenerateApi</mainClass> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 				<configuration> | ||||
| 					<mainClass>org.kar.karideo.WebLauncher</mainClass> | ||||
| 					<mainClass/> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<!-- Create the source bundle --> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-source-plugin</artifactId> | ||||
| 				<version>3.2.1</version> | ||||
| 				<version>4.0.0-beta-1</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>attach-sources</id> | ||||
| @@ -95,10 +150,12 @@ | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-surefire-plugin</artifactId> | ||||
| 				<version>3.0.0-M5</version> | ||||
| 				<version>3.2.5</version> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-assembly-plugin</artifactId> | ||||
| 				<version>3.7.1</version> | ||||
| 				<configuration> | ||||
| 					<archive> | ||||
| 						<manifest> | ||||
| @@ -110,94 +167,21 @@ | ||||
| 					</descriptorRefs> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<!-- Create coverage --> | ||||
| 			<!-- | ||||
| 			<plugin> | ||||
| 				<groupId>org.jacoco</groupId> | ||||
| 				<artifactId>jacoco-maven-plugin</artifactId> | ||||
| 				<version>0.8.10</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>prepare-agent</id> | ||||
| 						<goals> | ||||
| 							<goal>prepare-agent</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>report</id> | ||||
| 						<phase>test</phase> | ||||
| 						<goals> | ||||
| 							<goal>report</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>jacoco-check</id> | ||||
| 						<goals> | ||||
| 							<goal>check</goal> | ||||
| 						</goals> | ||||
| 						<configuration> | ||||
| 							<rules> | ||||
| 								<rule> | ||||
| 									<element>PACKAGE</element> | ||||
| 									<limits> | ||||
| 										<limit> | ||||
| 											<counter>LINE</counter> | ||||
| 											<value>COVEREDRATIO</value> | ||||
| 											<minimum>0.50</minimum> | ||||
| 										</limit> | ||||
| 									</limits> | ||||
| 								</rule> | ||||
| 							</rules> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 			</plugin> | ||||
| 			--> | ||||
| 			<!-- Java-doc generation for stand-alone site --> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-javadoc-plugin</artifactId> | ||||
| 				<version>3.2.0</version> | ||||
| 				<version>3.3.0</version> | ||||
| 				<configuration> | ||||
| 					<show>private</show> | ||||
| 					<nohelp>true</nohelp> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<groupId>org.codehaus.mojo</groupId> | ||||
| 				<artifactId>exec-maven-plugin</artifactId> | ||||
| 				<version>3.1.0</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>exec-application</id> | ||||
| 						<phase>package</phase> | ||||
| 						<goals> | ||||
| 							<goal>java</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 				<configuration> | ||||
| 					<mainClass>org.kar.karideo.WebLauncher</mainClass> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<!-- Check the style of the code --> | ||||
| 			<!-- | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-checkstyle-plugin</artifactId> | ||||
| 				<version>3.1.0</version> | ||||
| 				<configuration> | ||||
| 					<configLocation>CheckStyle.xml</configLocation> | ||||
| 					<consoleOutput>true</consoleOutput> | ||||
| 					<failOnViolation>true</failOnViolation> | ||||
| 					<failsOnError>true</failsOnError> | ||||
| 					<includeTestSourceDirectory>true</includeTestSourceDirectory> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<groupId>net.revelc.code.formatter</groupId> | ||||
| 				<artifactId>formatter-maven-plugin</artifactId> | ||||
| 				<version>2.12.2</version> | ||||
| 				<version>2.24.1</version> | ||||
| 				<configuration> | ||||
| 					<encoding>UTF-8</encoding> | ||||
| 					<lineEnding>LF</lineEnding> | ||||
| @@ -221,7 +205,15 @@ | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 			</plugin> | ||||
| 			--> | ||||
| 			<plugin> | ||||
| 				<groupId>com.github.spotbugs</groupId> | ||||
| 				<artifactId>spotbugs-maven-plugin</artifactId> | ||||
| 				<version>4.8.5.0</version> | ||||
| 				<configuration> | ||||
| 					<includeFilterFile>spotbugs-security-include.xml</includeFilterFile> | ||||
| 					<excludeFilterFile>spotbugs-security-exclude.xml</excludeFilterFile> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 		</plugins> | ||||
| 	</build> | ||||
| 	<!-- Generate Java-docs As Part Of Project Reports --> | ||||
| @@ -230,7 +222,7 @@ | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-javadoc-plugin</artifactId> | ||||
| 				<version>3.2.0</version> | ||||
| 				<version>3.3.0</version> | ||||
| 				<configuration> | ||||
| 					<show>public</show> | ||||
| 				</configuration> | ||||
|   | ||||
							
								
								
									
										245
									
								
								back/pom.xml.versionsBackup
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										245
									
								
								back/pom.xml.versionsBackup
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,245 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
| 	<modelVersion>4.0.0</modelVersion> | ||||
| 	<groupId>org.kar</groupId> | ||||
| 	<artifactId>karideo</artifactId> | ||||
| 	<version>0.3.0</version> | ||||
| 	<properties> | ||||
| 		<maven.compiler.version>3.1</maven.compiler.version> | ||||
| 		<maven.compiler.source>21</maven.compiler.source> | ||||
| 		<maven.compiler.target>21</maven.compiler.target> | ||||
| 		<maven.dependency.version>3.1.1</maven.dependency.version> | ||||
| 	</properties> | ||||
| 	<repositories> | ||||
| 		<repository> | ||||
| 			<id>gitea</id> | ||||
| 			<url>https://gitea.atria-soft.org/api/packages/kangaroo-and-rabbit/maven</url> | ||||
| 		</repository> | ||||
| 	</repositories> | ||||
| 	<dependencies> | ||||
| 		<dependency> | ||||
| 			<groupId>kangaroo-and-rabbit</groupId> | ||||
| 			<artifactId>archidata</artifactId> | ||||
| 			<version>0.11.0</version> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>org.slf4j</groupId> | ||||
| 			<artifactId>slf4j-simple</artifactId> | ||||
| 			<version>2.0.9</version> | ||||
| 		</dependency> | ||||
| 		<dependency> | ||||
| 			<groupId>com.fasterxml.jackson.datatype</groupId> | ||||
| 			<artifactId>jackson-datatype-jsr310</artifactId> | ||||
| 			<version>2.16.1</version> | ||||
| 		</dependency> | ||||
| 		<!-- | ||||
| 		************************************************************  | ||||
| 		** TEST dependency                                        ** | ||||
| 		************************************************************ | ||||
| 		--> | ||||
| 		<dependency> | ||||
| 			<groupId>org.junit.jupiter</groupId> | ||||
| 			<artifactId>junit-jupiter-api</artifactId> | ||||
| 			<version>5.10.1</version> | ||||
| 			<scope>test</scope> | ||||
| 		</dependency> | ||||
| 	    <dependency> | ||||
| 	        <groupId>org.junit.jupiter</groupId> | ||||
| 	        <artifactId>junit-jupiter-engine</artifactId> | ||||
| 			<version>5.10.1</version> | ||||
| 	        <scope>test</scope> | ||||
| 	    </dependency> | ||||
| 	</dependencies> | ||||
| 	<build> | ||||
| 		<sourceDirectory>src</sourceDirectory> | ||||
| 		<testSourceDirectory>test/src</testSourceDirectory> | ||||
| 		<directory>${project.basedir}/out/maven/</directory> | ||||
| 		<resources> | ||||
| 			<resource> | ||||
| 				<directory>src/resources</directory> | ||||
| 			</resource> | ||||
| 		</resources> | ||||
| 		<testResources> | ||||
| 			<testResource> | ||||
| 				<directory>${basedir}/test/resources</directory> | ||||
| 			</testResource> | ||||
| 		</testResources> | ||||
| 		<plugins> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-compiler-plugin</artifactId> | ||||
| 				<version>${maven.compiler.version}</version> | ||||
| 				<configuration> | ||||
| 					<source>${maven.compiler.source}</source> | ||||
| 					<target>${maven.compiler.target}</target> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<groupId>org.codehaus.mojo</groupId> | ||||
| 				<artifactId>exec-maven-plugin</artifactId> | ||||
| 				<version>1.4.0</version> | ||||
| 				<configuration> | ||||
| 					<mainClass>org.kar.karideo.WebLauncher</mainClass> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<!-- Create the source bundle --> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-source-plugin</artifactId> | ||||
| 				<version>3.2.1</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>attach-sources</id> | ||||
| 						<goals> | ||||
| 							<goal>jar</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 			</plugin> | ||||
| 			<!-- junit results --> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-surefire-plugin</artifactId> | ||||
| 				<version>3.0.0-M5</version> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<artifactId>maven-assembly-plugin</artifactId> | ||||
| 				<configuration> | ||||
| 					<archive> | ||||
| 						<manifest> | ||||
| 							<mainClass>fully.qualified.MainClass</mainClass> | ||||
| 						</manifest> | ||||
| 					</archive> | ||||
| 					<descriptorRefs> | ||||
| 						<descriptorRef>jar-with-dependencies</descriptorRef> | ||||
| 					</descriptorRefs> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<!-- Create coverage --> | ||||
| 			<!-- | ||||
| 			<plugin> | ||||
| 				<groupId>org.jacoco</groupId> | ||||
| 				<artifactId>jacoco-maven-plugin</artifactId> | ||||
| 				<version>0.8.10</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>prepare-agent</id> | ||||
| 						<goals> | ||||
| 							<goal>prepare-agent</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>report</id> | ||||
| 						<phase>test</phase> | ||||
| 						<goals> | ||||
| 							<goal>report</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 					<execution> | ||||
| 						<id>jacoco-check</id> | ||||
| 						<goals> | ||||
| 							<goal>check</goal> | ||||
| 						</goals> | ||||
| 						<configuration> | ||||
| 							<rules> | ||||
| 								<rule> | ||||
| 									<element>PACKAGE</element> | ||||
| 									<limits> | ||||
| 										<limit> | ||||
| 											<counter>LINE</counter> | ||||
| 											<value>COVEREDRATIO</value> | ||||
| 											<minimum>0.50</minimum> | ||||
| 										</limit> | ||||
| 									</limits> | ||||
| 								</rule> | ||||
| 							</rules> | ||||
| 						</configuration> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 			</plugin> | ||||
| 			--> | ||||
| 			<!-- Java-doc generation for stand-alone site --> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-javadoc-plugin</artifactId> | ||||
| 				<version>3.2.0</version> | ||||
| 				<configuration> | ||||
| 					<show>private</show> | ||||
| 					<nohelp>true</nohelp> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<groupId>org.codehaus.mojo</groupId> | ||||
| 				<artifactId>exec-maven-plugin</artifactId> | ||||
| 				<version>3.1.0</version> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<id>exec-application</id> | ||||
| 						<phase>package</phase> | ||||
| 						<goals> | ||||
| 							<goal>java</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 				<configuration> | ||||
| 					<mainClass>org.kar.karideo.WebLauncher</mainClass> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<!-- Check the style of the code --> | ||||
| 			<!-- | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-checkstyle-plugin</artifactId> | ||||
| 				<version>3.1.0</version> | ||||
| 				<configuration> | ||||
| 					<configLocation>CheckStyle.xml</configLocation> | ||||
| 					<consoleOutput>true</consoleOutput> | ||||
| 					<failOnViolation>true</failOnViolation> | ||||
| 					<failsOnError>true</failsOnError> | ||||
| 					<includeTestSourceDirectory>true</includeTestSourceDirectory> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 			<plugin> | ||||
| 				<groupId>net.revelc.code.formatter</groupId> | ||||
| 				<artifactId>formatter-maven-plugin</artifactId> | ||||
| 				<version>2.12.2</version> | ||||
| 				<configuration> | ||||
| 					<encoding>UTF-8</encoding> | ||||
| 					<lineEnding>LF</lineEnding> | ||||
| 					<configFile>Formatter.xml</configFile> | ||||
| 					<directories> | ||||
| 						<directory>src/</directory> | ||||
| 						<directory>test/src</directory> | ||||
| 					</directories> | ||||
| 					<includes> | ||||
| 						<include>**/*.java</include> | ||||
| 					</includes> | ||||
| 					<excludes> | ||||
| 						<exclude>module-info.java</exclude> | ||||
| 					</excludes> | ||||
| 				</configuration> | ||||
| 				<executions> | ||||
| 					<execution> | ||||
| 						<goals> | ||||
| 							<goal>validate</goal> | ||||
| 						</goals> | ||||
| 					</execution> | ||||
| 				</executions> | ||||
| 			</plugin> | ||||
| 			--> | ||||
| 		</plugins> | ||||
| 	</build> | ||||
| 	<!-- Generate Java-docs As Part Of Project Reports --> | ||||
| 	<reporting> | ||||
| 		<plugins> | ||||
| 			<plugin> | ||||
| 				<groupId>org.apache.maven.plugins</groupId> | ||||
| 				<artifactId>maven-javadoc-plugin</artifactId> | ||||
| 				<version>3.2.0</version> | ||||
| 				<configuration> | ||||
| 					<show>public</show> | ||||
| 				</configuration> | ||||
| 			</plugin> | ||||
| 		</plugins> | ||||
| 	</reporting> | ||||
| </project> | ||||
| @@ -1,9 +1,9 @@ | ||||
| org.kar.karideo.dataTmpFolder=/application/data/tmp | ||||
| org.kar.karideo.dataTmpFolder=/application/data/media | ||||
| org.kar.karideo.rest.oauth=http://192.168.1.156:21080/oauth/api/ | ||||
| org.kar.karideo.db.host=1992.156.1.156 | ||||
| org.kar.karideo.db.port=20306 | ||||
| org.kar.karideo.db.login=root | ||||
| org.kar.karideo.db.port=klkhj456gkgtkhjgvkujfhjgkjhgsdfhb3467465fgdhdesfgh | ||||
| org.kar.karideo.db.name=karideo | ||||
| org.kar.karideo.address=http://0.0.0.0:18080/karideo/api/ | ||||
| org.atriasoft.karideo.dataTmpFolder=/application/data/tmp | ||||
| org.atriasoft.karideo.dataTmpFolder=/application/data/media | ||||
| org.atriasoft.karideo.rest.oauth=http://192.168.1.156:21080/oauth/api/ | ||||
| org.atriasoft.karideo.db.host=1992.156.1.156 | ||||
| org.atriasoft.karideo.db.port=20306 | ||||
| org.atriasoft.karideo.db.login=root | ||||
| org.atriasoft.karideo.db.port=klkhj456gkgtkhjgvkujfhjgkjhgsdfhb3467465fgdhdesfgh | ||||
| org.atriasoft.karideo.db.name=karideo | ||||
| org.atriasoft.karideo.address=http://0.0.0.0:18080/karideo/api/ | ||||
|   | ||||
							
								
								
									
										17
									
								
								back/src/org/atriasoft/karideo/GenerateApi.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								back/src/org/atriasoft/karideo/GenerateApi.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| package org.atriasoft.karideo; | ||||
|  | ||||
| import org.atriasoft.karideo.migration.Initialization; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class GenerateApi { | ||||
| 	private final static Logger LOGGER = LoggerFactory.getLogger(GenerateApi.class); | ||||
|  | ||||
| 	private GenerateApi() {} | ||||
|  | ||||
| 	public static void main(final String[] args) throws Exception { | ||||
| 		LOGGER.info("Generate API"); | ||||
| 		Initialization.generateObjects(); | ||||
| 		LOGGER.info("STOP the REST server."); | ||||
| 	} | ||||
| } | ||||
| @@ -1,51 +1,53 @@ | ||||
| package org.kar.karideo; | ||||
| package org.atriasoft.karideo; | ||||
| 
 | ||||
| import java.net.URI; | ||||
| import java.util.TimeZone; | ||||
| import java.util.logging.LogManager; | ||||
| 
 | ||||
| import org.atriasoft.archidata.UpdateJwtPublicKey; | ||||
| import org.atriasoft.archidata.api.DataResource; | ||||
| import org.atriasoft.archidata.catcher.GenericCatcher; | ||||
| import org.atriasoft.archidata.cron.CronScheduler; | ||||
| import org.atriasoft.archidata.db.DbConfig; | ||||
| import org.atriasoft.archidata.filter.CORSFilter; | ||||
| import org.atriasoft.archidata.filter.OptionFilter; | ||||
| import org.atriasoft.archidata.migration.MigrationEngine; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.archidata.tools.ContextGenericTools; | ||||
| import org.atriasoft.karideo.api.Front; | ||||
| import org.atriasoft.karideo.api.HealthCheck; | ||||
| import org.atriasoft.karideo.api.MediaResource; | ||||
| import org.atriasoft.karideo.api.SeasonResource; | ||||
| import org.atriasoft.karideo.api.SeriesResource; | ||||
| import org.atriasoft.karideo.api.TypeResource; | ||||
| import org.atriasoft.karideo.api.UserMediaAdvancementResource; | ||||
| import org.atriasoft.karideo.api.UserResource; | ||||
| import org.atriasoft.karideo.filter.KarideoAuthenticationFilter; | ||||
| import org.atriasoft.karideo.job.BackupJob; | ||||
| import org.atriasoft.karideo.migration.Initialization; | ||||
| import org.glassfish.grizzly.http.server.HttpServer; | ||||
| import org.glassfish.jersey.grizzly2.httpserver.GrizzlyHttpServerFactory; | ||||
| import org.glassfish.jersey.jackson.JacksonFeature; | ||||
| import org.glassfish.jersey.media.multipart.MultiPartFeature; | ||||
| import org.glassfish.jersey.server.ResourceConfig; | ||||
| import org.kar.archidata.GlobalConfiguration; | ||||
| import org.kar.archidata.UpdateJwtPublicKey; | ||||
| import org.kar.archidata.api.DataResource; | ||||
| import org.kar.archidata.catcher.ExceptionCatcher; | ||||
| import org.kar.archidata.catcher.FailExceptionCatcher; | ||||
| import org.kar.archidata.catcher.InputExceptionCatcher; | ||||
| import org.kar.archidata.catcher.SystemExceptionCatcher; | ||||
| import org.kar.archidata.db.DBConfig; | ||||
| import org.kar.archidata.filter.CORSFilter; | ||||
| import org.kar.archidata.filter.OptionFilter; | ||||
| import org.kar.archidata.migration.MigrationEngine; | ||||
| import org.kar.archidata.tools.ConfigBaseVariable; | ||||
| import org.kar.karideo.api.Front; | ||||
| import org.kar.karideo.api.HealthCheck; | ||||
| import org.kar.karideo.api.MediaResource; | ||||
| import org.kar.karideo.api.SeasonResource; | ||||
| import org.kar.karideo.api.SeriesResource; | ||||
| import org.kar.karideo.api.TypeResource; | ||||
| import org.kar.karideo.api.UserMediaAdvancementResource; | ||||
| import org.kar.karideo.api.UserResource; | ||||
| import org.kar.karideo.filter.KarideoAuthenticationFilter; | ||||
| import org.kar.karideo.migration.Initialization; | ||||
| import org.kar.karideo.migration.Migration20230810; | ||||
| import org.kar.karideo.migration.Migration20231015; | ||||
| import org.kar.karideo.migration.Migration20231126; | ||||
| import org.kar.karideo.migration.Migration20240226; | ||||
| import org.glassfish.jersey.server.validation.ValidationFeature; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.slf4j.bridge.SLF4JBridgeHandler; | ||||
| 
 | ||||
| import jakarta.ws.rs.core.UriBuilder; | ||||
| 
 | ||||
| public class WebLauncher { | ||||
| 	final static Logger LOGGER = LoggerFactory.getLogger(WebLauncher.class); | ||||
| 	public static DBConfig dbConfig; | ||||
| 	protected UpdateJwtPublicKey keyUpdater = null; | ||||
| 	protected HttpServer server = null; | ||||
| 	protected CronScheduler scheduler = null; | ||||
| 
 | ||||
| 	public WebLauncher() { | ||||
| 		TimeZone.setDefault(TimeZone.getTimeZone("UTC")); | ||||
| 		ConfigBaseVariable.bdDatabase = "karideo"; | ||||
| 		this.scheduler = new CronScheduler(); | ||||
| 		this.scheduler.setGracePeriodMinutes(1); | ||||
| 	} | ||||
| 
 | ||||
| 	private static URI getBaseURI() { | ||||
| @@ -58,16 +60,17 @@ public class WebLauncher { | ||||
| 		WebLauncher.LOGGER.info("Add initialization"); | ||||
| 		migrationEngine.setInit(new Initialization()); | ||||
| 		WebLauncher.LOGGER.info("Add migration since last version"); | ||||
| 		migrationEngine.add(new Migration20230810()); | ||||
| 		migrationEngine.add(new Migration20231015()); | ||||
| 		migrationEngine.add(new Migration20231126()); | ||||
| 		migrationEngine.add(new Migration20240226()); | ||||
| 		// migrationEngine.add(new Migration20250928()); | ||||
| 		WebLauncher.LOGGER.info("Migrate the DB [START]"); | ||||
| 		migrationEngine.migrateWaitAdmin(GlobalConfiguration.dbConfig); | ||||
| 		migrationEngine.migrateWaitAdmin(new DbConfig()); | ||||
| 		WebLauncher.LOGGER.info("Migrate the DB [STOP]"); | ||||
| 	} | ||||
| 
 | ||||
| 	public static void main(final String[] args) throws Exception { | ||||
| 		// Loop-back of logger JDK logging API to SLF4J | ||||
| 		LogManager.getLogManager().reset(); | ||||
| 		SLF4JBridgeHandler.install(); | ||||
| 
 | ||||
| 		WebLauncher.LOGGER.info("[START] application wake UP"); | ||||
| 		final WebLauncher launcher = new WebLauncher(); | ||||
| 		launcher.migrateDB(); | ||||
| @@ -79,7 +82,7 @@ public class WebLauncher { | ||||
| 		WebLauncher.LOGGER.info("STOP the REST server"); | ||||
| 	} | ||||
| 
 | ||||
| 	public void process() throws InterruptedException { | ||||
| 	public void process() throws Exception { | ||||
| 
 | ||||
| 		// =================================================================== | ||||
| 		// Configure resources | ||||
| @@ -95,10 +98,7 @@ public class WebLauncher { | ||||
| 		// global authentication system | ||||
| 		rc.register(KarideoAuthenticationFilter.class); | ||||
| 		// register exception catcher | ||||
| 		rc.register(InputExceptionCatcher.class); | ||||
| 		rc.register(SystemExceptionCatcher.class); | ||||
| 		rc.register(FailExceptionCatcher.class); | ||||
| 		rc.register(ExceptionCatcher.class); | ||||
| 		GenericCatcher.addAll(rc); | ||||
| 		// add default resource: | ||||
| 		rc.register(UserResource.class); | ||||
| 		rc.register(SeriesResource.class); | ||||
| @@ -110,14 +110,20 @@ public class WebLauncher { | ||||
| 
 | ||||
| 		rc.register(HealthCheck.class); | ||||
| 		rc.register(Front.class); | ||||
| 
 | ||||
| 		ContextGenericTools.addJsr310(rc); | ||||
| 		// add jackson to be discover when we are ins stand-alone server | ||||
| 		rc.register(JacksonFeature.class); | ||||
| 		// enable jersey specific validations (@Valid) | ||||
| 		rc.register(ValidationFeature.class); | ||||
| 		// enable this to show low level request | ||||
| 		//rc.property(LoggingFeature.LOGGING_FEATURE_LOGGER_LEVEL_SERVER, Level.WARNING.getName()); | ||||
| 		// rc.property(LoggingFeature.LOGGING_FEATURE_LOGGER_LEVEL_SERVER, Level.WARNING.getName()); | ||||
| 
 | ||||
| 		this.server = GrizzlyHttpServerFactory.createHttpServer(getBaseURI(), rc); | ||||
| 		final HttpServer serverLink = this.server; | ||||
| 		// for (final NetworkListener listener : serverLink.getListeners()) { | ||||
| 		// listener.getKeepAlive().setIdleTimeoutInSeconds(30); // Set idle timeout | ||||
| 		// listener.getKeepAlive().setMaxRequestsCount(80); // Set request timeout | ||||
| 		// } | ||||
| 		Runtime.getRuntime().addShutdownHook(new Thread(new Runnable() { | ||||
| 			@Override | ||||
| 			public void run() { | ||||
| @@ -132,6 +138,14 @@ public class WebLauncher { | ||||
| 		this.keyUpdater = new UpdateJwtPublicKey(); | ||||
| 		this.keyUpdater.start(); | ||||
| 
 | ||||
| 		// =================================================================== | ||||
| 		// start generic scheduler ... | ||||
| 		// =================================================================== | ||||
| 		this.scheduler.addTask("backup", "0 0 * * *", new BackupJob()); | ||||
| 		this.scheduler.start(); | ||||
| 		// execute a backup before start... | ||||
| 		new BackupJob().run(); | ||||
| 
 | ||||
| 		// =================================================================== | ||||
| 		// run JERSEY | ||||
| 		// =================================================================== | ||||
| @@ -149,6 +163,10 @@ public class WebLauncher { | ||||
| 			this.server.shutdownNow(); | ||||
| 			this.server = null; | ||||
| 		} | ||||
| 		if (this.scheduler != null) { | ||||
| 			this.scheduler.stop(); | ||||
| 			this.scheduler = null; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	public void stopOther() { | ||||
							
								
								
									
										55
									
								
								back/src/org/atriasoft/karideo/WebLauncherLocal.java
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										55
									
								
								back/src/org/atriasoft/karideo/WebLauncherLocal.java
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,55 @@ | ||||
|  | ||||
| package org.atriasoft.karideo; | ||||
|  | ||||
| import java.util.logging.LogManager; | ||||
|  | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.karideo.migration.Initialization; | ||||
| import org.atriasoft.karideo.util.ConfigVariable; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.slf4j.bridge.SLF4JBridgeHandler; | ||||
|  | ||||
| public class WebLauncherLocal extends WebLauncher { | ||||
| 	private static final Logger LOGGER = LoggerFactory.getLogger(WebLauncherLocal.class); | ||||
|  | ||||
| 	private WebLauncherLocal() {} | ||||
|  | ||||
| 	public static void main(final String[] args) throws Exception { | ||||
| 		// Loop-back of logger JDK logging API to SLF4J | ||||
| 		LogManager.getLogManager().reset(); | ||||
| 		SLF4JBridgeHandler.install(); | ||||
| 		// Generate the APIs in type-script | ||||
| 		Initialization.generateObjects(); | ||||
| 		final WebLauncherLocal launcher = new WebLauncherLocal(); | ||||
| 		launcher.process(); | ||||
| 		LOGGER.info("end-configure the server & wait finish process:"); | ||||
| 		Thread.currentThread().join(); | ||||
| 		LOGGER.info("STOP the REST server:"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void process() throws Exception { | ||||
| 		if (true) { | ||||
| 			// for local test: | ||||
| 			ConfigBaseVariable.apiAdress = "http://0.0.0.0:18080/karideo/api/"; | ||||
| 			ConfigBaseVariable.testMode = "true"; | ||||
| 			ConfigBaseVariable.dbType = "mongo"; | ||||
| 		} | ||||
| 		if (ConfigVariable.isInitWithBackup()) { | ||||
| 			Initialization.initializeWithBackup(); | ||||
| 		} | ||||
| 		try { | ||||
| 			super.migrateDB(); | ||||
| 		} catch (final Exception e) { | ||||
| 			e.printStackTrace(); | ||||
| 			while (true) { | ||||
| 				LOGGER.error("============================================================================"); | ||||
| 				LOGGER.error("== Migration fail ==> waiting intervention of administrator..."); | ||||
| 				LOGGER.error("============================================================================"); | ||||
| 				Thread.sleep(60 * 60 * 1000); | ||||
| 			} | ||||
| 		} | ||||
| 		super.process(); | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										502
									
								
								back/src/org/atriasoft/karideo/api/DataResource.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										502
									
								
								back/src/org/atriasoft/karideo/api/DataResource.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,502 @@ | ||||
| package org.atriasoft.karideo.api; | ||||
|  | ||||
| import java.awt.Graphics2D; | ||||
| import java.awt.image.BufferedImage; | ||||
| import java.io.ByteArrayOutputStream; | ||||
| import java.io.File; | ||||
| import java.io.FileInputStream; | ||||
| import java.io.FileNotFoundException; | ||||
| import java.io.FileOutputStream; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| import java.io.OutputStream; | ||||
| import java.io.RandomAccessFile; | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.Paths; | ||||
| import java.nio.file.StandardCopyOption; | ||||
| import java.security.MessageDigest; | ||||
| import java.security.NoSuchAlgorithmException; | ||||
| import java.util.Date; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import javax.imageio.ImageIO; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiInputOptional; | ||||
| import org.atriasoft.archidata.annotation.security.PermitTokenInURI; | ||||
| import org.atriasoft.archidata.api.MediaStreamer; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.dataAccess.QueryCondition; | ||||
| import org.atriasoft.archidata.dataAccess.options.Condition; | ||||
| import org.atriasoft.archidata.exception.FailException; | ||||
| import org.atriasoft.archidata.filter.GenericContext; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.glassfish.jersey.media.multipart.FormDataContentDisposition; | ||||
| import org.glassfish.jersey.media.multipart.FormDataParam; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.HeaderParam; | ||||
| import jakarta.ws.rs.InternalServerErrorException; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.QueryParam; | ||||
| import jakarta.ws.rs.core.CacheControl; | ||||
| import jakarta.ws.rs.core.Context; | ||||
| import jakarta.ws.rs.core.HttpHeaders; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
| import jakarta.ws.rs.core.Response; | ||||
| import jakarta.ws.rs.core.SecurityContext; | ||||
| import jakarta.ws.rs.core.StreamingOutput; | ||||
|  | ||||
| // https://stackoverflow.com/questions/35367113/jersey-webservice-scalable-approach-to-download-file-and-reply-to-client | ||||
| // https://gist.github.com/aitoroses/4f7a2b197b732a6a691d | ||||
|  | ||||
| @Path("/data") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class DataResource { | ||||
| 	private static final Logger LOGGER = LoggerFactory.getLogger(DataResource.class); | ||||
| 	private final static int CHUNK_SIZE = 1024 * 1024; // 1MB chunks | ||||
| 	private final static int CHUNK_SIZE_IN = 50 * 1024 * 1024; // 1MB chunks | ||||
| 	/** Upload some datas */ | ||||
| 	private static long tmpFolderId = 1; | ||||
|  | ||||
| 	private static void createFolder(final String path) throws IOException { | ||||
| 		if (!Files.exists(Paths.get(path))) { | ||||
| 			Files.createDirectories(Paths.get(path)); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public static long getTmpDataId() { | ||||
| 		return tmpFolderId++; | ||||
| 	} | ||||
|  | ||||
| 	public static String getTmpFileInData(final long tmpFolderId) { | ||||
| 		final String filePath = ConfigBaseVariable.getTmpDataFolder() + File.separator + tmpFolderId; | ||||
| 		try { | ||||
| 			createFolder(ConfigBaseVariable.getTmpDataFolder() + File.separator); | ||||
| 		} catch (final IOException e) { | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return filePath; | ||||
| 	} | ||||
|  | ||||
| 	public static String getFileDataOld(final UUID uuid) { | ||||
| 		final String stringUUID = uuid.toString(); | ||||
| 		final String part1 = stringUUID.substring(0, 2); | ||||
| 		final String part2 = stringUUID.substring(2, 4); | ||||
| 		final String part3 = stringUUID.substring(4); | ||||
| 		final String finalPath = part1 + File.separator + part2; | ||||
| 		String filePath = ConfigBaseVariable.getMediaDataFolder() + "_uuid" + File.separator + finalPath + File.separator; | ||||
| 		try { | ||||
| 			createFolder(filePath); | ||||
| 		} catch (final IOException e) { | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		filePath += part3; | ||||
| 		return filePath; | ||||
| 	} | ||||
|  | ||||
| 	public static String getFileData(final ObjectId oid) { | ||||
| 		final String stringOid = oid.toHexString(); | ||||
| 		String dir1 = stringOid.substring(0, 2); | ||||
| 		String dir2 = stringOid.substring(2, 4); | ||||
| 		String dir3 = stringOid.substring(4, 6); | ||||
| 		try { | ||||
| 			final MessageDigest digest = MessageDigest.getInstance("SHA-256"); | ||||
| 			final byte[] hashBytes = digest.digest(oid.toByteArray()); | ||||
| 			dir1 = String.format("%02x", hashBytes[0]); | ||||
| 			dir2 = String.format("%02x", hashBytes[1]); | ||||
| 			dir3 = String.format("%02x", hashBytes[2]); | ||||
| 		} catch (final NoSuchAlgorithmException ex) { | ||||
| 			LOGGER.error("Fail to generate the hash of the objectId ==> ise direct value ... {}", ex.getMessage()); | ||||
| 		} | ||||
| 		final String finalPath = dir1 + File.separator + dir2 + File.separator + dir3; | ||||
| 		String filePath = ConfigBaseVariable.getMediaDataFolder() + "_oid" + File.separator + finalPath + File.separator; | ||||
| 		try { | ||||
| 			createFolder(filePath); | ||||
| 		} catch (final IOException e) { | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		filePath += stringOid; | ||||
| 		return filePath; | ||||
| 	} | ||||
|  | ||||
| 	public static String getFileMetaData(final ObjectId oid) { | ||||
| 		return getFileData(oid) + ".json"; | ||||
| 	} | ||||
|  | ||||
| 	public Data getWithSha512(final String sha512) { | ||||
| 		LOGGER.info("find sha512 = {}", sha512); | ||||
| 		try { | ||||
| 			return DataAccess.getWhere(Data.class, new Condition(new QueryCondition("sha512", "=", sha512))); | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	public Data getWithId(final long id) { | ||||
| 		LOGGER.info("find id = {}", id); | ||||
| 		try { | ||||
| 			return DataAccess.get(Data.class, id); | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	protected String getMimeType(final String extension) throws IOException { | ||||
| 		return switch (extension.toLowerCase()) { | ||||
| 		case "jpg", "jpeg" -> "image/jpeg"; | ||||
| 		case "png" -> "image/png"; | ||||
| 		case "webp" -> "image/webp"; | ||||
| 		case "mka" -> "audio/x-matroska"; | ||||
| 		case "mkv" -> "video/x-matroska"; | ||||
| 		case "webm" -> "video/webm"; | ||||
| 		default -> throw new IOException("Can not find the mime type of data input: '" + extension + "'"); | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| 	public Data createNewData(final long tmpUID, final String originalFileName, final String sha512) throws IOException { | ||||
| 		// determine mime type: | ||||
| 		Data injectedData = new Data(); | ||||
| 		String mimeType = ""; | ||||
| 		final String extension = originalFileName.substring(originalFileName.lastIndexOf('.') + 1); | ||||
| 		mimeType = getMimeType(extension); | ||||
| 		injectedData.mimeType = mimeType; | ||||
| 		injectedData.sha512 = sha512; | ||||
| 		final String tmpPath = getTmpFileInData(tmpUID); | ||||
| 		injectedData.size = Files.size(Paths.get(tmpPath)); | ||||
|  | ||||
| 		try { | ||||
| 			injectedData = DataAccess.insert(injectedData); | ||||
| 		} catch (final Exception e) { | ||||
| 			e.printStackTrace(); | ||||
| 			return null; | ||||
| 		} | ||||
| 		final String mediaPath = getFileData(injectedData.oid); | ||||
| 		LOGGER.info("src = {}", tmpPath); | ||||
| 		LOGGER.info("dst = {}", mediaPath); | ||||
| 		Files.move(Paths.get(tmpPath), Paths.get(mediaPath), StandardCopyOption.ATOMIC_MOVE); | ||||
| 		LOGGER.info("Move done"); | ||||
| 		return injectedData; | ||||
| 	} | ||||
|  | ||||
| 	public static void modeFileOldModelToNewModel(final UUID uuid, final ObjectId oid) throws IOException { | ||||
| 		String mediaCurentPath = getFileDataOld(uuid); | ||||
| 		String mediaDestPath = getFileData(oid); | ||||
| 		LOGGER.info("src = {}", mediaCurentPath); | ||||
| 		LOGGER.info("dst = {}", mediaDestPath); | ||||
| 		if (Files.exists(Paths.get(mediaCurentPath))) { | ||||
| 			LOGGER.info("move: {} ==> {}", mediaCurentPath, mediaDestPath); | ||||
| 			Files.move(Paths.get(mediaCurentPath), Paths.get(mediaDestPath), StandardCopyOption.ATOMIC_MOVE); | ||||
| 		} | ||||
| 		// Move old meta-data... | ||||
| 		mediaCurentPath = mediaCurentPath.substring(mediaCurentPath.length() - 4) + "meta.json"; | ||||
| 		mediaDestPath = mediaCurentPath.substring(mediaDestPath.length() - 4) + "meta.json"; | ||||
| 		if (Files.exists(Paths.get(mediaCurentPath))) { | ||||
| 			LOGGER.info("moveM: {} ==> {}", mediaCurentPath, mediaDestPath); | ||||
| 			Files.move(Paths.get(mediaCurentPath), Paths.get(mediaDestPath), StandardCopyOption.ATOMIC_MOVE); | ||||
| 		} | ||||
| 		LOGGER.info("Move done"); | ||||
| 	} | ||||
|  | ||||
| 	public static String saveTemporaryFile(final InputStream uploadedInputStream, final long idData) throws FailException { | ||||
| 		return saveFile(uploadedInputStream, DataResource.getTmpFileInData(idData)); | ||||
| 	} | ||||
|  | ||||
| 	public static void removeTemporaryFile(final long idData) { | ||||
| 		final String filepath = DataResource.getTmpFileInData(idData); | ||||
| 		if (Files.exists(Paths.get(filepath))) { | ||||
| 			try { | ||||
| 				Files.delete(Paths.get(filepath)); | ||||
| 			} catch (final IOException e) { | ||||
| 				LOGGER.info("can not delete temporary file : {}", Paths.get(filepath)); | ||||
| 				e.printStackTrace(); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// save uploaded file to a defined location on the server | ||||
| 	static String saveFile(final InputStream uploadedInputStream, final String serverLocation) throws FailException { | ||||
| 		String out = ""; | ||||
| 		MessageDigest md = null; | ||||
| 		try (OutputStream outpuStream = new FileOutputStream(new File(serverLocation))) { | ||||
| 			md = MessageDigest.getInstance("SHA-512"); | ||||
| 			outpuStream.flush(); | ||||
| 		} catch (final IOException ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Can not write in temporary file", ex); | ||||
| 		} catch (final NoSuchAlgorithmException ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Can not find sha512 algorithms", ex); | ||||
| 		} | ||||
| 		if (md != null) { | ||||
| 			try (OutputStream outpuStream = new FileOutputStream(new File(serverLocation))) { | ||||
| 				int read = 0; | ||||
| 				final byte[] bytes = new byte[CHUNK_SIZE_IN]; | ||||
| 				while ((read = uploadedInputStream.read(bytes)) != -1) { | ||||
| 					// logger.info("write {}", read); | ||||
| 					md.update(bytes, 0, read); | ||||
| 					outpuStream.write(bytes, 0, read); | ||||
| 				} | ||||
| 				LOGGER.info("Flush input stream ... {}", serverLocation); | ||||
| 				outpuStream.flush(); | ||||
| 				// create the end of sha512 | ||||
| 				final byte[] sha512Digest = md.digest(); | ||||
| 				// convert in hexadecimal | ||||
| 				out = bytesToHex(sha512Digest); | ||||
| 				uploadedInputStream.close(); | ||||
| 			} catch (final IOException ex) { | ||||
| 				throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Can not write in temporary file", ex); | ||||
| 			} | ||||
| 		} | ||||
| 		return out; | ||||
| 	} | ||||
|  | ||||
| 	public static String bytesToHex(final byte[] bytes) { | ||||
| 		final StringBuilder sb = new StringBuilder(); | ||||
| 		for (final byte b : bytes) { | ||||
| 			sb.append(String.format("%02x", b)); | ||||
| 		} | ||||
| 		return sb.toString(); | ||||
| 	} | ||||
|  | ||||
| 	public Data getSmall(final ObjectId oid) { | ||||
| 		try { | ||||
| 			return DataAccess.get(Data.class, oid); | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	@POST | ||||
| 	@Path("/upload/") | ||||
| 	@Consumes({ MediaType.MULTIPART_FORM_DATA }) | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Insert a new data in the data environment", tags = "SYSTEM") | ||||
| 	public void uploadFile(@Context final SecurityContext sc, @FormDataParam("file") final InputStream fileInputStream, @FormDataParam("file") final FormDataContentDisposition fileMetaData) | ||||
| 			throws FailException { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		LOGGER.info("==================================================="); | ||||
| 		LOGGER.info("== DATA uploadFile {}", (gc == null ? "null" : gc.userByToken)); | ||||
| 		LOGGER.info("==================================================="); | ||||
| 		// public NodeSmall uploadFile(final FormDataMultiPart form) { | ||||
| 		LOGGER.info("Upload file: "); | ||||
| 		final String filePath = ConfigBaseVariable.getTmpDataFolder() + File.separator + tmpFolderId++; | ||||
| 		try { | ||||
| 			createFolder(ConfigBaseVariable.getTmpDataFolder() + File.separator); | ||||
| 		} catch (final IOException ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Impossible to create the folder in the server", ex); | ||||
| 		} | ||||
| 		saveFile(fileInputStream, filePath); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{oid}") | ||||
| 	@PermitTokenInURI | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Produces(MediaType.APPLICATION_OCTET_STREAM) | ||||
| 	@Operation(description = "Get back some data from the data environment", tags = "SYSTEM") | ||||
| 	public Response retrieveDataId(@Context final SecurityContext sc, @QueryParam(HttpHeaders.AUTHORIZATION) final String token, @HeaderParam("Range") final String range, | ||||
| 			@PathParam("oid") final ObjectId oid) throws FailException { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		// logger.info("==================================================="); | ||||
| 		LOGGER.info("== DATA retrieveDataId ? oid={} user={}", oid, (gc == null ? "null" : gc.userByToken)); | ||||
| 		// logger.info("==================================================="); | ||||
| 		final Data value = getSmall(oid); | ||||
| 		if (value == null) { | ||||
| 			return Response.status(404).entity("media NOT FOUND: " + oid).type("text/plain").build(); | ||||
| 		} | ||||
| 		try { | ||||
| 			return buildStream(getFileData(oid), range, value.mimeType == null ? "application/octet-stream" : value.mimeType); | ||||
| 		} catch (final Exception ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Fail to build output stream", ex); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("thumbnail/{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@PermitTokenInURI | ||||
| 	@Produces(MediaType.APPLICATION_OCTET_STREAM) | ||||
| 	@Operation(description = "Get a thumbnail of from the data environment (if resize is possible)", tags = "SYSTEM") | ||||
| 	// @CacheMaxAge(time = 10, unit = TimeUnit.DAYS) | ||||
| 	public Response retrieveDataThumbnailId(@Context final SecurityContext sc, @QueryParam(HttpHeaders.AUTHORIZATION) final String token, @HeaderParam("Range") final String range, | ||||
| 			@PathParam("oid") final ObjectId oid) throws FailException { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		LOGGER.info("==================================================="); | ||||
| 		LOGGER.info("== DATA retrieveDataThumbnailId ? {}", (gc == null ? "null" : gc.userByToken)); | ||||
| 		LOGGER.info("==================================================="); | ||||
| 		final Data value = getSmall(oid); | ||||
| 		if (value == null) { | ||||
| 			return Response.status(404).entity("media NOT FOUND: " + oid).type("text/plain").build(); | ||||
| 		} | ||||
| 		final String filePathName = getFileData(oid); | ||||
| 		final File inputFile = new File(filePathName); | ||||
| 		if (!inputFile.exists()) { | ||||
| 			return Response.status(404).entity("{\"error\":\"media Does not exist: " + oid + "\"}").type("application/json").build(); | ||||
| 		} | ||||
| 		if (value.mimeType.contentEquals("image/jpeg") || value.mimeType.contentEquals("image/png") | ||||
| 		// || value.mimeType.contentEquals("image/webp") | ||||
| 		) { | ||||
| 			// reads input image | ||||
| 			BufferedImage inputImage; | ||||
| 			try { | ||||
| 				inputImage = ImageIO.read(inputFile); | ||||
| 			} catch (final IOException ex) { | ||||
| 				throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Fail to READ the image", ex); | ||||
| 			} | ||||
| 			LOGGER.info("input size image: {}x{} type={}", inputImage.getWidth(), inputImage.getHeight(), inputImage.getType()); | ||||
| 			final int scaledWidth = ConfigBaseVariable.getThumbnailWidth(); | ||||
| 			final int scaledHeight = (int) ((float) inputImage.getHeight() / (float) inputImage.getWidth() * scaledWidth); | ||||
| 			// creates output image | ||||
| 			final BufferedImage outputImage = new BufferedImage(scaledWidth, scaledHeight, inputImage.getType()); | ||||
|  | ||||
| 			// scales the input image to the output image | ||||
| 			final Graphics2D g2d = outputImage.createGraphics(); | ||||
| 			LOGGER.info("output size image: {}x{}", scaledWidth, scaledHeight); | ||||
| 			g2d.drawImage(inputImage, 0, 0, scaledWidth, scaledHeight, null); | ||||
| 			g2d.dispose(); | ||||
| 			// create the output stream: | ||||
| 			final ByteArrayOutputStream baos = new ByteArrayOutputStream(); | ||||
| 			try { | ||||
| 				ImageIO.write(outputImage, ConfigBaseVariable.getThumbnailFormat(), baos); | ||||
| 			} catch (final IOException e) { | ||||
| 				e.printStackTrace(); | ||||
| 				return Response.status(500).entity("Internal Error: resize fail: " + e.getMessage()).type("text/plain").build(); | ||||
| 			} | ||||
| 			final byte[] imageData = baos.toByteArray(); | ||||
| 			LOGGER.info("output length {}", imageData.length); | ||||
| 			if (imageData.length == 0) { | ||||
| 				LOGGER.error("Fail to convert image... Availlable format:"); | ||||
| 				for (final String data : ImageIO.getWriterFormatNames()) { | ||||
| 					LOGGER.error("    - {}", data); | ||||
| 				} | ||||
| 			} | ||||
| 			final Response.ResponseBuilder out = Response.ok(imageData).header(HttpHeaders.CONTENT_LENGTH, imageData.length); | ||||
| 			try { | ||||
| 				out.type(getMimeType(ConfigBaseVariable.getThumbnailFormat())); | ||||
| 			} catch (final IOException ex) { | ||||
| 				throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Fail to convert mime type of " + ConfigBaseVariable.getThumbnailFormat(), ex); | ||||
| 			} | ||||
| 			// TODO: move this in a decorator !!! | ||||
| 			final CacheControl cc = new CacheControl(); | ||||
| 			cc.setMaxAge(3600); | ||||
| 			cc.setNoCache(false); | ||||
| 			out.cacheControl(cc); | ||||
| 			return out.build(); | ||||
| 		} | ||||
| 		try { | ||||
| 			return buildStream(filePathName, range, value.mimeType); | ||||
| 		} catch (final Exception ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Fail to build output stream", ex); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{oid}/{name}") | ||||
| 	@PermitTokenInURI | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Produces(MediaType.APPLICATION_OCTET_STREAM) | ||||
| 	@Operation(description = "Get back some data from the data environment (with a beautiful name (permit download with basic name)", tags = "SYSTEM") | ||||
| 	public Response retrieveDataFull(@Context final SecurityContext sc, @QueryParam(HttpHeaders.AUTHORIZATION) final String token, @ApiInputOptional @HeaderParam("Range") final String range, | ||||
| 			@PathParam("oid") final ObjectId oid, @PathParam("name") final String name) throws Exception { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		// logger.info("==================================================="); | ||||
| 		LOGGER.info("== DATA retrieveDataFull ? id={} user={}", oid, (gc == null ? "null" : gc.userByToken)); | ||||
| 		// logger.info("==================================================="); | ||||
| 		final Data value = getSmall(oid); | ||||
| 		if (value == null) { | ||||
| 			return Response.status(404).entity("media NOT FOUND: " + oid).type("text/plain").build(); | ||||
| 		} | ||||
| 		return buildStream(getFileData(oid), range, value.mimeType == null ? "application/octet-stream" : value.mimeType); | ||||
| 	} | ||||
|  | ||||
| 	/** Adapted from http://stackoverflow.com/questions/12768812/video-streaming-to-ipad-does-not-work-with-tapestry5/12829541#12829541 | ||||
| 	 * | ||||
| 	 * @param range range header | ||||
| 	 * @return Streaming output | ||||
| 	 * @throws FileNotFoundException | ||||
| 	 * @throws Exception IOException if an error occurs in streaming. */ | ||||
| 	private Response buildStream(final String filename, final String range, final String mimeType) throws FailException { | ||||
| 		final File file = new File(filename); | ||||
| 		// logger.info("request range : {}", range); | ||||
| 		// range not requested : Firefox does not send range headers | ||||
| 		if (range == null) { | ||||
| 			final StreamingOutput output = new StreamingOutput() { | ||||
| 				@Override | ||||
| 				public void write(final OutputStream out) { | ||||
| 					try (FileInputStream in = new FileInputStream(file)) { | ||||
| 						final byte[] buf = new byte[1024 * 1024]; | ||||
| 						int len; | ||||
| 						while ((len = in.read(buf)) != -1) { | ||||
| 							try { | ||||
| 								out.write(buf, 0, len); | ||||
| 								out.flush(); | ||||
| 								// logger.info("---- wrote {} bytes file ----", len); | ||||
| 							} catch (final IOException ex) { | ||||
| 								LOGGER.info("remote close connection"); | ||||
| 								break; | ||||
| 							} | ||||
| 						} | ||||
| 					} catch (final IOException ex) { | ||||
| 						throw new InternalServerErrorException(ex); | ||||
| 					} | ||||
| 				} | ||||
| 			}; | ||||
| 			final Response.ResponseBuilder out = Response.ok(output).header(HttpHeaders.CONTENT_LENGTH, file.length()); | ||||
| 			if (mimeType != null) { | ||||
| 				out.type(mimeType); | ||||
| 			} | ||||
| 			return out.build(); | ||||
|  | ||||
| 		} | ||||
|  | ||||
| 		final String[] ranges = range.split("=")[1].split("-"); | ||||
| 		final long from = Long.parseLong(ranges[0]); | ||||
|  | ||||
| 		// logger.info("request range : {}", ranges.length); | ||||
| 		// Chunk media if the range upper bound is unspecified. Chrome, Opera sends "bytes=0-" | ||||
| 		long to = CHUNK_SIZE + from; | ||||
| 		if (ranges.length == 1) { | ||||
| 			to = file.length() - 1; | ||||
| 		} else if (to >= file.length()) { | ||||
| 			to = file.length() - 1; | ||||
| 		} | ||||
| 		final String responseRange = String.format("bytes %d-%d/%d", from, to, file.length()); | ||||
| 		// LOGGER.info("responseRange: {}", responseRange); | ||||
| 		try { | ||||
| 			final RandomAccessFile raf = new RandomAccessFile(file, "r"); | ||||
| 			raf.seek(from); | ||||
|  | ||||
| 			final long len = to - from + 1; | ||||
| 			final MediaStreamer streamer = new MediaStreamer(len, raf); | ||||
| 			final Response.ResponseBuilder out = Response.ok(streamer).status(Response.Status.PARTIAL_CONTENT).header("Accept-Ranges", "bytes").header("Content-Range", responseRange) | ||||
| 					.header(HttpHeaders.CONTENT_LENGTH, streamer.getLenth()).header(HttpHeaders.LAST_MODIFIED, new Date(file.lastModified())); | ||||
| 			if (mimeType != null) { | ||||
| 				out.type(mimeType); | ||||
| 			} | ||||
| 			return out.build(); | ||||
| 		} catch (final FileNotFoundException ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Fail to find the required file.", ex); | ||||
| 		} catch (final IOException ex) { | ||||
| 			throw new FailException(Response.Status.INTERNAL_SERVER_ERROR, "Fail to access to the required file.", ex); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public void undelete(final Long id) throws Exception { | ||||
| 		DataAccess.unsetDelete(Data.class, id); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,16 +1,15 @@ | ||||
| package org.kar.karideo.api; | ||||
| package org.atriasoft.karideo.api; | ||||
| 
 | ||||
| 
 | ||||
| import org.kar.archidata.api.FrontGeneric; | ||||
| import org.atriasoft.archidata.api.FrontGeneric; | ||||
| 
 | ||||
| import jakarta.ws.rs.Path; | ||||
| 
 | ||||
| import org.kar.karideo.util.ConfigVariable; | ||||
| import org.atriasoft.karideo.util.ConfigVariable; | ||||
| 
 | ||||
| @Path("/front") | ||||
| public class Front extends FrontGeneric { | ||||
| 	public Front() { | ||||
| 		this.baseFrontFolder = ConfigVariable.getFrontFolder(); | ||||
| 		 | ||||
| 
 | ||||
| 	} | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| package org.kar.karideo.api; | ||||
| package org.atriasoft.karideo.api; | ||||
| 
 | ||||
| import org.kar.archidata.exception.FailException; | ||||
| import org.kar.archidata.tools.ConfigBaseVariable; | ||||
| import org.kar.archidata.tools.JWTWrapper; | ||||
| import org.atriasoft.archidata.exception.FailException; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.archidata.tools.JWTWrapper; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| 
 | ||||
| @@ -1,36 +1,38 @@ | ||||
| package org.kar.karideo.api; | ||||
| package org.atriasoft.karideo.api; | ||||
| 
 | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| import java.sql.SQLException; | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
| 
 | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiTypeScriptProgress; | ||||
| import org.atriasoft.archidata.annotation.checker.GroupUpdate; | ||||
| import org.atriasoft.archidata.annotation.checker.ValidGroup; | ||||
| import org.atriasoft.archidata.api.DataResource; | ||||
| import org.atriasoft.archidata.dataAccess.DBAccess; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.exception.FailException; | ||||
| import org.atriasoft.archidata.exception.InputException; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.tools.DataTools; | ||||
| import org.atriasoft.karideo.model.Media; | ||||
| import org.atriasoft.karideo.model.Season; | ||||
| import org.atriasoft.karideo.model.Series; | ||||
| import org.atriasoft.karideo.model.Type; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.glassfish.jersey.media.multipart.FormDataContentDisposition; | ||||
| import org.glassfish.jersey.media.multipart.FormDataParam; | ||||
| import org.kar.archidata.annotation.AsyncType; | ||||
| import org.kar.archidata.annotation.TypeScriptProgress; | ||||
| import org.kar.archidata.api.DataResource; | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.dataAccess.addOn.AddOnDataJson; | ||||
| import org.kar.archidata.exception.FailException; | ||||
| import org.kar.archidata.exception.InputException; | ||||
| import org.kar.archidata.model.Data; | ||||
| import org.kar.archidata.tools.DataTools; | ||||
| import org.kar.karideo.model.Media; | ||||
| import org.kar.karideo.model.Season; | ||||
| import org.kar.karideo.model.Series; | ||||
| import org.kar.karideo.model.Type; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| 
 | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.validation.Valid; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.PATCH; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.PUT; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| @@ -56,15 +58,16 @@ public class MediaResource { | ||||
| 		return DataAccess.get(Media.class, id); | ||||
| 	} | ||||
| 
 | ||||
| 	@PATCH | ||||
| 	@Path("{id}") | ||||
| 	@PUT | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific Media", tags = "GLOBAL") | ||||
| 	public Media patch(@PathParam("id") final Long id, @AsyncType(Media.class) final String jsonRequest) throws Exception { | ||||
| 		LOGGER.info("update video {} ==> '{}'", id, jsonRequest); | ||||
| 		DataAccess.updateWithJson(Media.class, id, jsonRequest); | ||||
| 		return DataAccess.get(Media.class, id); | ||||
| 	public Media put(@PathParam("oid") final ObjectId oid, @Valid @ValidGroup(GroupUpdate.class) final Media media) throws Exception { | ||||
| 		LOGGER.info("update video {} ==> '{}'", oid, media); | ||||
| 		media.oid = oid; | ||||
| 		DataAccess.update(media, oid); | ||||
| 		return DataAccess.get(Media.class, oid); | ||||
| 	} | ||||
| 
 | ||||
| 	private String multipartCorrection(final String data) { | ||||
| @@ -77,6 +80,9 @@ public class MediaResource { | ||||
| 		if (data.contentEquals("null")) { | ||||
| 			return null; | ||||
| 		} | ||||
| 		if (data.contentEquals("undefined")) { | ||||
| 			return null; | ||||
| 		} | ||||
| 		return data; | ||||
| 	} | ||||
| 
 | ||||
| @@ -84,35 +90,44 @@ public class MediaResource { | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes({ MediaType.MULTIPART_FORM_DATA }) | ||||
| 	@Operation(description = "Create a new Media", tags = "GLOBAL") | ||||
| 	@TypeScriptProgress | ||||
| 	public Media uploadFile( // | ||||
| 			@FormDataParam("fileName") String fileName, // | ||||
| 			@FormDataParam("universe") String universe, // | ||||
| 			@FormDataParam("series") String series, // | ||||
| 			//@FormDataParam("seriesId") String seriesId, // Not used ... | ||||
| 			@FormDataParam("season") String season, // | ||||
| 			@FormDataParam("episode") String episode, // | ||||
| 			@FormDataParam("title") String title, // | ||||
| 	@ApiTypeScriptProgress | ||||
| 	public Media uploadMedia( // | ||||
| 			// @AsyncType(Long.class) @FormDataParam("universeId") String universeId, // | ||||
| 			// @AsyncType(Long.class) @FormDataParam("typeId") String typeId, // | ||||
| 			// @AsyncType(Long.class) @FormDataParam("seriesId") String seriesId, // | ||||
| 			// @AsyncType(Long.class) @FormDataParam("season") String season, // value of the season ==> local add if needed | ||||
| 			// @AsyncType(Long.class) @FormDataParam("episode") String episode, // value of the season ==> local add if needed | ||||
| 			@FormDataParam("universeId") String universeId, // | ||||
| 			@FormDataParam("typeId") String typeId, // | ||||
| 			@FormDataParam("seriesId") String seriesId, // | ||||
| 			@FormDataParam("season") String season, // value of the season ==> local add if needed | ||||
| 			@FormDataParam("episode") String episode, // value of the season ==> local add if needed | ||||
| 			@FormDataParam("title") String title, // | ||||
| 			@FormDataParam("file") final InputStream fileInputStream, // | ||||
| 			@FormDataParam("file") final FormDataContentDisposition fileMetaData // | ||||
| 			) throws FailException { | ||||
| 		try { | ||||
| 	) throws FailException { | ||||
| 		try (DBAccess db = DBAccess.createInterface()) { | ||||
| 			// correct input string stream : | ||||
| 			fileName = multipartCorrection(fileName); | ||||
| 			universe = multipartCorrection(universe); | ||||
| 			series = multipartCorrection(series); | ||||
| 			final String fileName = multipartCorrection(fileMetaData.getFileName()); | ||||
| 			universeId = multipartCorrection(universeId); | ||||
| 			final ObjectId universeOid = universeId != null ? new ObjectId(universeId) : null; | ||||
| 			typeId = multipartCorrection(typeId); | ||||
| 			final ObjectId typeOid = typeId != null ? new ObjectId(typeId) : null; | ||||
| 			seriesId = multipartCorrection(seriesId); | ||||
| 			final ObjectId seriesOid = seriesId != null ? new ObjectId(seriesId) : null; | ||||
| 			season = multipartCorrection(season); | ||||
| 			final Long seasonLong = season != null ? Long.parseLong(season) : null; | ||||
| 			episode = multipartCorrection(episode); | ||||
| 			title = multipartCorrection(title); | ||||
| 			typeId = multipartCorrection(typeId); | ||||
| 
 | ||||
| 			//public NodeSmall uploadFile(final FormDataMultiPart form) { | ||||
| 			// todo: check if all remotes Id exist ... | ||||
| 
 | ||||
| 			// public NodeSmall uploadFile(final FormDataMultiPart form) { | ||||
| 			LOGGER.info("Upload media file: {}", fileMetaData); | ||||
| 			LOGGER.info("    - fileName: {}", fileName); | ||||
| 			LOGGER.info("    - universe: {}", universe); | ||||
| 			LOGGER.info("    - series: {}", series); | ||||
| 			LOGGER.info("    - season: {}", season); | ||||
| 			LOGGER.info("    - universe: {}", universeOid); | ||||
| 			LOGGER.info("    - series: {}", seriesOid); | ||||
| 			LOGGER.info("    - season: {}", seasonLong); | ||||
| 			LOGGER.info("    - episode: {}", episode); | ||||
| 			LOGGER.info("    - title: {}", title); | ||||
| 			LOGGER.info("    - type: {}", typeId); | ||||
| @@ -125,21 +140,21 @@ public class MediaResource { | ||||
| 
 | ||||
| 			final long tmpUID = DataResource.getTmpDataId(); | ||||
| 			final String sha512 = DataResource.saveTemporaryFile(fileInputStream, tmpUID); | ||||
| 			Data data = DataResource.getWithSha512(sha512); | ||||
| 			Data data = DataTools.getWithSha512(db, sha512); | ||||
| 			if (data == null) { | ||||
| 				LOGGER.info("Need to add the data in the BDD ... "); | ||||
| 				System.out.flush(); | ||||
| 				try { | ||||
| 					data = DataResource.createNewData(tmpUID, fileName, sha512); | ||||
| 					data = DataTools.createNewData(db, tmpUID, fileName, sha512); | ||||
| 				} catch (final IOException ex) { | ||||
| 					DataResource.removeTemporaryFile(tmpUID); | ||||
| 					ex.printStackTrace(); | ||||
| 					throw new FailException("can not create input media (the data model has an internal error"); | ||||
| 				} | ||||
| 			} else if (data!= null && data.deleted != null && data.deleted) { | ||||
| 			} else if (data != null && data.deleted != null && data.deleted) { | ||||
| 				LOGGER.info("Data already exist but deleted"); | ||||
| 				System.out.flush(); | ||||
| 				DataTools.undelete(data.id); | ||||
| 				DataTools.undelete(db, data.oid); | ||||
| 				data.deleted = false; | ||||
| 			} else { | ||||
| 				LOGGER.info("Data already exist ... all good"); | ||||
| @@ -148,29 +163,35 @@ public class MediaResource { | ||||
| 			// Fist step: retieve all the Id of each parents:... | ||||
| 			LOGGER.info("Find typeNode"); | ||||
| 			// check if id of type exist: | ||||
| 			final Type typeNode = TypeResource.getId(Long.parseLong(typeId)); | ||||
| 			final Type typeNode = TypeResource.getId(typeOid); | ||||
| 			if (typeNode == null) { | ||||
| 				DataResource.removeTemporaryFile(tmpUID); | ||||
| 				throw new InputException("typeId", "TypeId does not exist ..."); | ||||
| 			} | ||||
| 			LOGGER.info("    ==> {}", typeNode); | ||||
| 			LOGGER.info("Find seriesNode"); | ||||
| 			// get uid of group: | ||||
| 			// check if id of type exist: | ||||
| 			Series seriesNode = null; | ||||
| 			if (series != null) { | ||||
| 				seriesNode = SeriesResource.getOrCreate(series, typeNode.id); | ||||
| 			if (seriesOid != null) { | ||||
| 				seriesNode = SeriesResource.getId(seriesOid); | ||||
| 				if (seriesNode == null) { | ||||
| 					DataResource.removeTemporaryFile(tmpUID); | ||||
| 					throw new InputException("seriesId", "seriesId does not exist ..."); | ||||
| 				} | ||||
| 				if (!seriesNode.typeId.equals(typeNode.oid)) { | ||||
| 					DataResource.removeTemporaryFile(tmpUID); | ||||
| 					throw new InputException("seriesId", "seriesId object have not the correct parent... '" + seriesNode.typeId + "' != '" + typeNode.oid + "'"); | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			LOGGER.info("    ==> {}", seriesNode); | ||||
| 			LOGGER.info("Find seasonNode"); | ||||
| 			// get uid of season: | ||||
| 			Season seasonNode = null; | ||||
| 			if (seriesNode == null && season != null) { | ||||
| 			if (seriesNode == null && seasonLong != null) { | ||||
| 				DataResource.removeTemporaryFile(tmpUID); | ||||
| 				throw new InputException("season", "Season is set but no seraies is set !!"); | ||||
| 				throw new InputException("season", "Season is set but no series is set !!"); | ||||
| 			} | ||||
| 			if (season != null) { | ||||
| 				seasonNode = SeasonResource.getOrCreate(season, seriesNode.id); | ||||
| 				seasonNode = SeasonResource.getOrCreate(season, seriesNode.oid); | ||||
| 			} | ||||
| 
 | ||||
| 			LOGGER.info("    ==> {}", seasonNode); | ||||
| @@ -179,21 +200,21 @@ public class MediaResource { | ||||
| 			try { | ||||
| 				final Media media = new Media(); | ||||
| 				media.name = title; | ||||
| 				media.dataId = data.id; | ||||
| 				media.typeId = typeNode.id; | ||||
| 				media.dataId = data.oid; | ||||
| 				media.typeId = typeNode.oid; | ||||
| 				media.seriesId = null; | ||||
| 				if (seriesNode != null) { | ||||
| 					media.seriesId = seriesNode.id; | ||||
| 					media.seriesId = seriesNode.oid; | ||||
| 				} | ||||
| 				media.seasonId = null; | ||||
| 				if (seasonNode != null) { | ||||
| 					media.seasonId = seasonNode.id; | ||||
| 					media.seasonId = seasonNode.oid; | ||||
| 				} | ||||
| 				media.episode = null; | ||||
| 				if (episode != null && !episode.contentEquals("")) { | ||||
| 					media.episode = Integer.parseInt(episode); | ||||
| 				} | ||||
| 				final Media out = DataAccess.insert(media); | ||||
| 				final Media out = db.insert(media); | ||||
| 				LOGGER.info("Generate new media {}", out); | ||||
| 				return out; | ||||
| 			} catch (final SQLException ex) { | ||||
| @@ -210,40 +231,11 @@ public class MediaResource { | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	@POST | ||||
| 	@Path("{id}/cover") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes({ MediaType.MULTIPART_FORM_DATA }) | ||||
| 	@AsyncType(Media.class) | ||||
| 	@Operation(description = "Upload a new season cover media", tags = "GLOBAL") | ||||
| 	@TypeScriptProgress | ||||
| 	public Media uploadCover( // | ||||
| 			@PathParam("id") final Long id, // | ||||
| 			@FormDataParam("fileName") final String fileName, // | ||||
| 			@FormDataParam("file") final InputStream fileInputStream, // | ||||
| 			@FormDataParam("file") final FormDataContentDisposition fileMetaData// | ||||
| 			) throws Exception { | ||||
| 		DataTools.uploadCover(Media.class, id, fileName, fileInputStream, fileMetaData); | ||||
| 		return DataAccess.get(Media.class, id); | ||||
| 	} | ||||
| 
 | ||||
| 	@DELETE | ||||
| 	@Path("{id}/cover/{coverId}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific cover of a media", tags = "GLOBAL") | ||||
| 	public Media removeCover( // | ||||
| 			@PathParam("id") final Long id, // | ||||
| 			@PathParam("coverId") final UUID coverId // | ||||
| 			) throws Exception { | ||||
| 		AddOnDataJson.removeLink(Media.class, id, "covers", coverId); | ||||
| 		return DataAccess.get(Media.class, id); | ||||
| 	} | ||||
| 
 | ||||
| 	@DELETE | ||||
| 	@Path("{id}") | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific Media", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("id") final Long id) throws Exception { | ||||
| 	public void remove(@PathParam("oid") final ObjectId id) throws Exception { | ||||
| 		DataAccess.delete(Media.class, id); | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										97
									
								
								back/src/org/atriasoft/karideo/api/SeasonResource.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								back/src/org/atriasoft/karideo/api/SeasonResource.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| package org.atriasoft.karideo.api; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.checker.GroupCreate; | ||||
| import org.atriasoft.archidata.annotation.checker.GroupUpdate; | ||||
| import org.atriasoft.archidata.annotation.checker.ValidGroup; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.dataAccess.QueryAnd; | ||||
| import org.atriasoft.archidata.dataAccess.QueryCondition; | ||||
| import org.atriasoft.archidata.dataAccess.options.Condition; | ||||
| import org.atriasoft.karideo.model.Season; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.validation.Valid; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.PUT; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
|  | ||||
| @Path("/season") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class SeasonResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(SeasonResource.class); | ||||
|  | ||||
| 	@GET | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get a specific Season with his ID", tags = "GLOBAL") | ||||
| 	public List<Season> gets() throws Exception { | ||||
| 		return DataAccess.gets(Season.class); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Get all season", tags = "GLOBAL") | ||||
| 	public Season get(@PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		return DataAccess.get(Season.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	/* ============================================================================= ADMIN SECTION: ============================================================================= */ | ||||
|  | ||||
| 	@POST | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Create a new season", tags = "GLOBAL") | ||||
| 	public Season post(@Valid @ValidGroup(GroupCreate.class) final Season season) throws Exception { | ||||
| 		return DataAccess.insert(season); | ||||
| 	} | ||||
|  | ||||
| 	@PUT | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific season", tags = "GLOBAL") | ||||
| 	public Season put(@PathParam("oid") final ObjectId oid, @Valid @ValidGroup(GroupUpdate.class) final Season season) throws Exception { | ||||
| 		season.oid = oid; | ||||
| 		DataAccess.update(season, oid); | ||||
| 		return DataAccess.get(Season.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific season", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		DataAccess.delete(Season.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	public static Season getOrCreate(final String name, final ObjectId seriesId) { | ||||
| 		try { | ||||
| 			Season out = DataAccess.getWhere(Season.class, new Condition(new QueryAnd(new QueryCondition("name", "=", name), new QueryCondition("seriesId", "=", seriesId)))); | ||||
| 			if (out == null) { | ||||
| 				out = new Season(); | ||||
| 				out.name = name; | ||||
| 				out.seriesId = seriesId; | ||||
| 				out = DataAccess.insert(out); | ||||
| 			} | ||||
| 			return out; | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										100
									
								
								back/src/org/atriasoft/karideo/api/SeriesResource.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								back/src/org/atriasoft/karideo/api/SeriesResource.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| package org.atriasoft.karideo.api; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.checker.GroupCreate; | ||||
| import org.atriasoft.archidata.annotation.checker.GroupUpdate; | ||||
| import org.atriasoft.archidata.annotation.checker.ValidGroup; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.dataAccess.QueryAnd; | ||||
| import org.atriasoft.archidata.dataAccess.QueryCondition; | ||||
| import org.atriasoft.archidata.dataAccess.options.Condition; | ||||
| import org.atriasoft.karideo.model.Series; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.validation.Valid; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.PUT; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
|  | ||||
| @Path("/series") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class SeriesResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(SeriesResource.class); | ||||
|  | ||||
| 	@GET | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get all Series", tags = "GLOBAL") | ||||
| 	public List<Series> gets() throws Exception { | ||||
| 		return DataAccess.gets(Series.class); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Get a specific Series with his ID", tags = "GLOBAL") | ||||
| 	public Series get(@PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		return DataAccess.get(Series.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	public static Series getId(final ObjectId oid) throws Exception { | ||||
| 		return DataAccess.get(Series.class, oid); | ||||
| 	} | ||||
| 	/* ============================================================================= ADMIN SECTION: ============================================================================= */ | ||||
|  | ||||
| 	@POST | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Create a new Series", tags = "GLOBAL") | ||||
| 	public Series post(@Valid @ValidGroup(GroupCreate.class) final Series jsonRequest) throws Exception { | ||||
| 		return DataAccess.insert(jsonRequest); | ||||
| 	} | ||||
|  | ||||
| 	@PUT | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific Series", tags = "GLOBAL") | ||||
| 	public Series put(@PathParam("oid") final ObjectId oid, @Valid @ValidGroup(GroupUpdate.class) final Series series) throws Exception { | ||||
| 		series.oid = oid; | ||||
| 		DataAccess.update(series, oid); | ||||
| 		return DataAccess.get(Series.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific Series", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		DataAccess.delete(Series.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	public static Series getOrCreate(final String name, final ObjectId typeId) { | ||||
| 		try { | ||||
| 			Series out = DataAccess.getWhere(Series.class, new Condition(new QueryAnd(new QueryCondition("name", "=", name), new QueryCondition("typeId", "=", typeId)))); | ||||
| 			if (out == null) { | ||||
| 				out = new Series(); | ||||
| 				out.name = name; | ||||
| 				out.typeId = typeId; | ||||
| 				out = DataAccess.insert(out); | ||||
| 			} | ||||
| 			return out; | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										99
									
								
								back/src/org/atriasoft/karideo/api/TypeResource.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								back/src/org/atriasoft/karideo/api/TypeResource.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | ||||
| package org.atriasoft.karideo.api; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.checker.GroupCreate; | ||||
| import org.atriasoft.archidata.annotation.checker.GroupUpdate; | ||||
| import org.atriasoft.archidata.annotation.checker.ValidGroup; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.dataAccess.QueryCondition; | ||||
| import org.atriasoft.archidata.dataAccess.options.Condition; | ||||
| import org.atriasoft.karideo.model.Type; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.validation.Valid; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.PUT; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
|  | ||||
| @Path("/type") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class TypeResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(TypeResource.class); | ||||
|  | ||||
| 	@GET | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get all Type", tags = "GLOBAL") | ||||
| 	public List<Type> gets() throws Exception { | ||||
| 		return DataAccess.gets(Type.class); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Get a specific Type with his ID", tags = "GLOBAL") | ||||
| 	public Type get(@PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		return DataAccess.get(Type.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	public static Type getId(final ObjectId oid) throws Exception { | ||||
| 		return DataAccess.get(Type.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	/* ============================================================================= ADMIN SECTION: ============================================================================= */ | ||||
|  | ||||
| 	@POST | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Create a new Type", tags = "GLOBAL") | ||||
| 	public Type post(@Valid @ValidGroup(GroupCreate.class) final Type jsonRequest) throws Exception { | ||||
| 		return DataAccess.insert(jsonRequest); | ||||
| 	} | ||||
|  | ||||
| 	@PUT | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific Type", tags = "GLOBAL") | ||||
| 	public Type patch(@PathParam("oid") final ObjectId oid, @Valid @ValidGroup(GroupUpdate.class) final Type type) throws Exception { | ||||
| 		type.oid = oid; | ||||
| 		DataAccess.update(type, oid); | ||||
| 		return DataAccess.get(Type.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific Type", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		DataAccess.delete(Type.class, oid); | ||||
| 	} | ||||
|  | ||||
| 	public static Type getOrCreate(final String name) { | ||||
| 		try { | ||||
| 			Type out = DataAccess.getWhere(Type.class, new Condition(new QueryCondition("name", "=", name))); | ||||
| 			if (out == null) { | ||||
| 				out = new Type(); | ||||
| 				out.name = name; | ||||
| 				out = DataAccess.insert(out); | ||||
| 			} | ||||
| 			return out; | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,22 +1,27 @@ | ||||
| package org.kar.karideo.api; | ||||
| package org.atriasoft.karideo.api; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.dataAccess.QueryAnd; | ||||
| import org.kar.archidata.dataAccess.QueryCondition; | ||||
| import org.kar.archidata.dataAccess.options.Condition; | ||||
| import org.kar.archidata.filter.GenericContext; | ||||
| import org.kar.karideo.model.UserMediaAdvancement; | ||||
| import org.atriasoft.archidata.annotation.checker.GroupUpdate; | ||||
| import org.atriasoft.archidata.annotation.checker.ValidGroup; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.dataAccess.QueryAnd; | ||||
| import org.atriasoft.archidata.dataAccess.QueryCondition; | ||||
| import org.atriasoft.archidata.dataAccess.options.Condition; | ||||
| import org.atriasoft.archidata.dataAccess.options.FilterValue; | ||||
| import org.atriasoft.archidata.filter.GenericContext; | ||||
| import org.atriasoft.karideo.model.UserMediaAdvancement; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| 
 | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.validation.Valid; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.PATCH; | ||||
| import jakarta.ws.rs.PUT; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| @@ -30,12 +35,12 @@ public class UserMediaAdvancementResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(UserMediaAdvancementResource.class); | ||||
| 
 | ||||
| 	@GET | ||||
| 	@Path("{id}") | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get a specific user advancement with his ID", tags = "GLOBAL") | ||||
| 	public UserMediaAdvancement get(@Context final SecurityContext sc, @PathParam("id") final Long id) throws Exception { | ||||
| 	public UserMediaAdvancement get(@Context final SecurityContext sc, @PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		return DataAccess.getWhere(UserMediaAdvancement.class, new Condition(new QueryAnd(new QueryCondition("mediaId", "=", id), new QueryCondition("userId", "=", gc.userByToken.id)))); | ||||
| 		return DataAccess.getWhere(UserMediaAdvancement.class, new Condition(new QueryAnd(new QueryCondition("mediaId", "=", oid), new QueryCondition("userId", "=", gc.userByToken.oid)))); | ||||
| 	} | ||||
| 
 | ||||
| 	@GET | ||||
| @@ -43,51 +48,45 @@ public class UserMediaAdvancementResource { | ||||
| 	@Operation(description = "Get all user advancement", tags = "GLOBAL") | ||||
| 	public List<UserMediaAdvancement> gets(@Context final SecurityContext sc) throws Exception { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		return DataAccess.getsWhere(UserMediaAdvancement.class, new Condition(new QueryCondition("userId", "=", gc.userByToken.id))); | ||||
| 		return DataAccess.getsWhere(UserMediaAdvancement.class, new Condition(new QueryCondition("userId", "=", gc.userByToken.oid))); | ||||
| 	} | ||||
| 
 | ||||
| 	/* ============================================================================= | ||||
| 	 * Modification SECTION: | ||||
| 	 * ============================================================================= */ | ||||
| 	/* ============================================================================= Modification SECTION: ============================================================================= */ | ||||
| 
 | ||||
| 	public record MediaInformations( | ||||
| 			int time, | ||||
| 			float percent, | ||||
| 			int count) {} | ||||
| 	public record MediaInformations(int time, float percent, int count) { | ||||
| 	} | ||||
| 
 | ||||
| 	//@POST | ||||
| 	//@Path("{id}") | ||||
| 	//@RolesAllowed("USER") | ||||
| 	//@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	public UserMediaAdvancement post(@Context final SecurityContext sc, @PathParam("id") final Long id, final MediaInformations data) throws Exception { | ||||
| 	// @POST | ||||
| 	// @Path("{id}") | ||||
| 	// @RolesAllowed("USER") | ||||
| 	// @Consumes(MediaType.APPLICATION_JSON) | ||||
| 	public UserMediaAdvancement post(@Context final SecurityContext sc, @PathParam("oid") final ObjectId oid, final MediaInformations data) throws Exception { | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		final UserMediaAdvancement elem = new UserMediaAdvancement(); | ||||
| 		elem.userId = gc.userByToken.id; | ||||
| 		elem.mediaId = id; | ||||
| 		elem.userId = gc.userByToken.oid; | ||||
| 		elem.mediaId = oid; | ||||
| 		elem.time = data.time; | ||||
| 		elem.percent = data.percent; | ||||
| 		elem.count = data.count; | ||||
| 		return DataAccess.insert(elem); | ||||
| 	} | ||||
| 
 | ||||
| 	public record MediaInformationsDelta( | ||||
| 			int time, | ||||
| 			float percent, | ||||
| 			boolean addCount) {} | ||||
| 	public record MediaInformationsDelta(int time, float percent, boolean addCount) { | ||||
| 	} | ||||
| 
 | ||||
| 	@PATCH | ||||
| 	@Path("{id}") | ||||
| 	@PUT | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a user advancement", tags = "GLOBAL") | ||||
| 	public UserMediaAdvancement patch(@Context final SecurityContext sc, @PathParam("id") final Long id, final MediaInformationsDelta data) throws Exception { | ||||
| 		final UserMediaAdvancement elem = get(sc, id); | ||||
| 	public UserMediaAdvancement put(@Context final SecurityContext sc, @PathParam("oid") final ObjectId oid, @Valid @ValidGroup(GroupUpdate.class) final MediaInformationsDelta data) throws Exception { | ||||
| 		final UserMediaAdvancement elem = get(sc, oid); | ||||
| 		if (elem == null) { | ||||
| 			// insert element | ||||
| 			if (data.addCount) { | ||||
| 				return post(sc, id, new MediaInformations(data.time(), data.percent(), 1)); | ||||
| 				return post(sc, oid, new MediaInformations(data.time(), data.percent(), 1)); | ||||
| 			} else { | ||||
| 				return post(sc, id, new MediaInformations(data.time(), data.percent(), 0)); | ||||
| 				return post(sc, oid, new MediaInformations(data.time(), data.percent(), 0)); | ||||
| 			} | ||||
| 		} | ||||
| 		elem.time = data.time; | ||||
| @@ -96,17 +95,17 @@ public class UserMediaAdvancementResource { | ||||
| 			elem.count++; | ||||
| 		} | ||||
| 		LOGGER.info("{},{},{}", elem.time, elem.percent, elem.count); | ||||
| 		final int nbAfected = DataAccess.update(elem, elem.id, List.of("time", "percent", "count")); | ||||
| 		return DataAccess.get(UserMediaAdvancement.class, elem.id); | ||||
| 		DataAccess.update(elem, elem.oid, new FilterValue("time", "percent", "count")); | ||||
| 		return DataAccess.get(UserMediaAdvancement.class, elem.oid); | ||||
| 	} | ||||
| 
 | ||||
| 	@DELETE | ||||
| 	@Path("{id}") | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Remove a specific user advancement", tags = "GLOBAL") | ||||
| 	public void remove(@Context final SecurityContext sc, @PathParam("id") final Long id) throws Exception { | ||||
| 		final UserMediaAdvancement elem = get(sc, id); | ||||
| 		DataAccess.delete(UserMediaAdvancement.class, elem.id); | ||||
| 	public void remove(@Context final SecurityContext sc, @PathParam("oid") final ObjectId oid) throws Exception { | ||||
| 		final UserMediaAdvancement elem = get(sc, oid); | ||||
| 		DataAccess.delete(UserMediaAdvancement.class, elem.oid); | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
| @@ -1,10 +1,11 @@ | ||||
| package org.kar.karideo.api; | ||||
| package org.atriasoft.karideo.api; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.filter.GenericContext; | ||||
| import org.kar.karideo.model.UserKarideo; | ||||
| import org.atriasoft.archidata.dataAccess.DataAccess; | ||||
| import org.atriasoft.archidata.filter.GenericContext; | ||||
| import org.atriasoft.karideo.model.UserKarideo; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| 
 | ||||
| @@ -27,11 +28,11 @@ public class UserResource { | ||||
| 
 | ||||
| 	@JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| 	public class UserOut { | ||||
| 		public long id; | ||||
| 		public ObjectId oid; | ||||
| 		public String login; | ||||
| 
 | ||||
| 		public UserOut(final long id, final String login) { | ||||
| 			this.id = id; | ||||
| 		public UserOut(final ObjectId oid, final String login) { | ||||
| 			this.oid = oid; | ||||
| 			this.login = login; | ||||
| 		} | ||||
| 
 | ||||
| @@ -56,10 +57,10 @@ public class UserResource { | ||||
| 
 | ||||
| 	// curl http://localhost:9993/api/users/3 | ||||
| 	@GET | ||||
| 	@Path("{id}") | ||||
| 	@Path("{oid}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Get a specific user data", tags = "SYSTEM") | ||||
| 	public UserKarideo get(@Context final SecurityContext sc, @PathParam("id") final long userId) { | ||||
| 	public UserKarideo get(@Context final SecurityContext sc, @PathParam("oid") final ObjectId userId) { | ||||
| 		System.out.println("getUser " + userId); | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		System.out.println("==================================================="); | ||||
| @@ -82,7 +83,7 @@ public class UserResource { | ||||
| 		LOGGER.debug("getMe()"); | ||||
| 		final GenericContext gc = (GenericContext) sc.getUserPrincipal(); | ||||
| 		LOGGER.debug("== USER ? {}", gc.userByToken); | ||||
| 		return new UserOut(gc.userByToken.id, gc.userByToken.name); | ||||
| 		return new UserOut(gc.userByToken.oid, gc.userByToken.name); | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| package org.kar.karideo.filter; | ||||
| package org.atriasoft.karideo.filter; | ||||
| 
 | ||||
| import org.kar.archidata.filter.AuthenticationFilter; | ||||
| import org.atriasoft.archidata.filter.AuthenticationFilter; | ||||
| 
 | ||||
| import jakarta.ws.rs.Priorities; | ||||
| import jakarta.ws.rs.ext.Provider; | ||||
| @@ -15,9 +15,9 @@ import jakarta.annotation.Priority; | ||||
| @Priority(Priorities.AUTHENTICATION) | ||||
| public class KarideoAuthenticationFilter extends AuthenticationFilter { | ||||
| 	final Logger logger = LoggerFactory.getLogger(KarideoAuthenticationFilter.class); | ||||
| 	 | ||||
|     public KarideoAuthenticationFilter() { | ||||
| 
 | ||||
| 	public KarideoAuthenticationFilter() { | ||||
| 		super("karideo"); | ||||
| 	} | ||||
|      | ||||
| 
 | ||||
| } | ||||
							
								
								
									
										60
									
								
								back/src/org/atriasoft/karideo/internal/Log.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								back/src/org/atriasoft/karideo/internal/Log.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| package org.atriasoft.karideo.internal; | ||||
|  | ||||
| //import io.scenarium.logger.LogLevel; | ||||
| //import io.scenarium.logger.Logger; | ||||
|  | ||||
| public class Log { | ||||
| 	// private static final String LIB_NAME = "logger"; | ||||
| 	// private static final String LIB_NAME_DRAW = Logger.getDrawableName(LIB_NAME); | ||||
| 	// private static final boolean PRINT_CRITICAL = Logger.getNeedPrint(LIB_NAME, LogLevel.CRITICAL); | ||||
| 	// private static final boolean PRINT_ERROR = Logger.getNeedPrint(LIB_NAME, LogLevel.ERROR); | ||||
| 	// private static final boolean PRINT_WARNING = Logger.getNeedPrint(LIB_NAME, LogLevel.WARNING); | ||||
| 	// private static final boolean PRINT_INFO = Logger.getNeedPrint(LIB_NAME, LogLevel.INFO); | ||||
| 	// private static final boolean PRINT_DEBUG = Logger.getNeedPrint(LIB_NAME, LogLevel.DEBUG); | ||||
| 	// private static final boolean PRINT_VERBOSE = Logger.getNeedPrint(LIB_NAME, LogLevel.VERBOSE); | ||||
| 	// private static final boolean PRINT_TODO = Logger.getNeedPrint(LIB_NAME, LogLevel.TODO); | ||||
| 	// private static final boolean PRINT_PRINT = Logger.getNeedPrint(LIB_NAME, LogLevel.PRINT); | ||||
| 	// | ||||
| 	// private Log() {} | ||||
| 	// | ||||
| 	// public static void print(String data) { | ||||
| 	// if (PRINT_PRINT) | ||||
| 	// Logger.print(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void todo(String data) { | ||||
| 	// if (PRINT_TODO) | ||||
| 	// Logger.todo(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void critical(String data) { | ||||
| 	// if (PRINT_CRITICAL) | ||||
| 	// Logger.critical(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void error(String data) { | ||||
| 	// if (PRINT_ERROR) | ||||
| 	// Logger.error(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void warning(String data) { | ||||
| 	// if (PRINT_WARNING) | ||||
| 	// Logger.warning(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void info(String data) { | ||||
| 	// if (PRINT_INFO) | ||||
| 	// Logger.info(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void debug(String data) { | ||||
| 	// if (PRINT_DEBUG) | ||||
| 	// Logger.debug(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
| 	// | ||||
| 	// public static void verbose(String data) { | ||||
| 	// if (PRINT_VERBOSE) | ||||
| 	// Logger.verbose(LIB_NAME_DRAW, data); | ||||
| 	// } | ||||
|  | ||||
| } | ||||
							
								
								
									
										47
									
								
								back/src/org/atriasoft/karideo/job/BackupJob.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								back/src/org/atriasoft/karideo/job/BackupJob.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| package org.atriasoft.karideo.job; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.Path; | ||||
| import java.nio.file.Paths; | ||||
| import java.time.ZonedDateTime; | ||||
| import java.time.format.DateTimeFormatter; | ||||
|  | ||||
| import org.atriasoft.archidata.backup.BackupEngine; | ||||
| import org.atriasoft.archidata.backup.BackupEngine.EngineBackupType; | ||||
| import org.atriasoft.archidata.exception.DataAccessException; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.karideo.migration.Initialization; | ||||
| import org.atriasoft.karideo.util.ConfigVariable; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class BackupJob implements Runnable { | ||||
|  | ||||
| 	final static Logger LOGGER = LoggerFactory.getLogger(BackupJob.class); | ||||
| 	BackupEngine engine; | ||||
|  | ||||
| 	public BackupJob() throws DataAccessException, IOException { | ||||
| 		final Path path = Paths.get(ConfigVariable.getBackupFolder()); | ||||
| 		Files.createDirectories(path); | ||||
| 		this.engine = new BackupEngine(path, ConfigBaseVariable.getDBName(), EngineBackupType.JSON_EXTENDED); | ||||
| 		this.engine.setEnableStoreOrRestoreData(false); | ||||
| 		for (final Class<?> clazz : Initialization.CLASSES_BASE) { | ||||
| 			this.engine.addClass(clazz); | ||||
| 		} | ||||
| 		this.engine.addCollection("counters"); | ||||
| 		this.engine.addCollection("KAR_migration"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void run() { | ||||
| 		LOGGER.warn("Backup request"); | ||||
| 		try { | ||||
| 			final String timestampUtc = ZonedDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd_HH:mm:ss.SSS")); | ||||
| 			this.engine.store(timestampUtc + "_full"); | ||||
| 		} catch (final Exception ex) { | ||||
| 			LOGGER.error("Fail in Backup: {}", ex.getMessage()); | ||||
| 			ex.printStackTrace(); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										131
									
								
								back/src/org/atriasoft/karideo/migration/Initialization.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								back/src/org/atriasoft/karideo/migration/Initialization.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,131 @@ | ||||
| package org.atriasoft.karideo.migration; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.Path; | ||||
| import java.nio.file.Paths; | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.api.DataResource; | ||||
| import org.atriasoft.archidata.backup.BackupEngine; | ||||
| import org.atriasoft.archidata.backup.BackupEngine.EngineBackupType; | ||||
| import org.atriasoft.archidata.dataAccess.DBAccess; | ||||
| import org.atriasoft.archidata.db.DbConfig; | ||||
| import org.atriasoft.archidata.exception.DataAccessException; | ||||
| import org.atriasoft.archidata.exception.FailException; | ||||
| import org.atriasoft.archidata.externalRestApi.AnalyzeApi; | ||||
| import org.atriasoft.archidata.externalRestApi.TsGenerateApi; | ||||
| import org.atriasoft.archidata.filter.PartRight; | ||||
| import org.atriasoft.archidata.migration.MigrationSqlStep; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.User; | ||||
| import org.atriasoft.archidata.model.token.JwtToken; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.karideo.api.Front; | ||||
| import org.atriasoft.karideo.api.HealthCheck; | ||||
| import org.atriasoft.karideo.api.MediaResource; | ||||
| import org.atriasoft.karideo.api.SeasonResource; | ||||
| import org.atriasoft.karideo.api.SeriesResource; | ||||
| import org.atriasoft.karideo.api.TypeResource; | ||||
| import org.atriasoft.karideo.api.UserMediaAdvancementResource; | ||||
| import org.atriasoft.karideo.api.UserResource; | ||||
| import org.atriasoft.karideo.model.Media; | ||||
| import org.atriasoft.karideo.model.Season; | ||||
| import org.atriasoft.karideo.model.Series; | ||||
| import org.atriasoft.karideo.model.Type; | ||||
| import org.atriasoft.karideo.model.UserMediaAdvancement; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import jakarta.ws.rs.InternalServerErrorException; | ||||
|  | ||||
| public class Initialization extends MigrationSqlStep { | ||||
| 	private static final Logger LOGGER = LoggerFactory.getLogger(Initialization.class); | ||||
|  | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
|  | ||||
| 	public static final List<Class<?>> CLASSES_BASE = List.of(Data.class, Media.class, Type.class, Series.class, Season.class, User.class, UserMediaAdvancement.class); | ||||
|  | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "Initialization"; | ||||
| 	} | ||||
|  | ||||
| 	public static void removeDB() throws DataAccessException, InternalServerErrorException, IOException { | ||||
| 		final DbConfig config = new DbConfig(); | ||||
| 		LOGGER.info("Remove DB '{}'", config.getDbName()); | ||||
| 		try (DBAccess dba = DBAccess.createInterface(config)) { | ||||
| 			dba.deleteDB(ConfigBaseVariable.bdDatabase); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** Restore backup file (./init/backup.tar.gz). */ | ||||
| 	public static void initializeWithBackup() throws IOException, DataAccessException, FailException { | ||||
| 		final Path path = Paths.get("./init/backup.tar.gz"); | ||||
| 		if (!Files.exists(path)) { | ||||
| 			throw new FailException("file: ./init/backup.tar.gz does not exist"); | ||||
| 		} | ||||
| 		removeDB(); | ||||
| 		final BackupEngine engine = new BackupEngine(Paths.get("."), ConfigBaseVariable.bdDatabase, EngineBackupType.JSON_EXTENDED); | ||||
| 		if (!engine.restoreFile(path, null)) { | ||||
| 			throw new FailException("Can not retrieve db from backup"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public static void generateObjects() throws Exception { | ||||
| 		LOGGER.info("Generate APIs"); | ||||
| 		final List<Class<?>> listOfResources = List.of(Front.class, HealthCheck.class, SeasonResource.class, SeriesResource.class, TypeResource.class, UserMediaAdvancementResource.class, | ||||
| 				UserResource.class, MediaResource.class, DataResource.class); | ||||
| 		final AnalyzeApi api = new AnalyzeApi(); | ||||
| 		api.addAllApi(listOfResources); | ||||
| 		api.addModel(JwtToken.class); | ||||
| 		api.addModel(PartRight.class); | ||||
| 		TsGenerateApi.generateApi(api, Paths.get("../front/src/back-api/")); | ||||
| 		LOGGER.info("Generate APIs (DONE)"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		for (final Class<?> clazz : CLASSES_BASE) { | ||||
| 			addClass(clazz); | ||||
| 		} | ||||
| 		addAction((final DBAccess da) -> { | ||||
| 			final List<Type> data = List.of(// | ||||
| 					new Type("Documentary", "Documentary (animals, space, earth...)"), // | ||||
| 					new Type("Movie", "Movie with real humans (film)"), // | ||||
| 					new Type("Animation", "Animation movies (film)"), // | ||||
| 					new Type("Short movie", "Small movies (less 2 minutes)"), // | ||||
| 					new Type("TV show", "TV show for old peoples"), // | ||||
| 					new Type("Animation TV show", "TV show for young peoples"), // | ||||
| 					new Type("Theater", "Theater play"), // | ||||
| 					new Type("One man show", "Recorded stand up"), // | ||||
| 					new Type("Concert", "Recorded concert"), // | ||||
| 					new Type("Opera", "Recorded opera") // | ||||
| 			); | ||||
| 			da.insertMultiple(data); | ||||
| 		}); | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	public static void dropAll(final DBAccess da) { | ||||
| 		for (final Class<?> element : CLASSES_BASE) { | ||||
| 			try { | ||||
| 				da.drop(element); | ||||
| 			} catch (final Exception ex) { | ||||
| 				LOGGER.error("Fail to drop table !!!!!!"); | ||||
| 				ex.printStackTrace(); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public static void cleanAll(final DBAccess da) { | ||||
| 		for (final Class<?> element : CLASSES_BASE) { | ||||
| 			try { | ||||
| 				da.cleanAll(element); | ||||
| 			} catch (final Exception ex) { | ||||
| 				LOGGER.error("Fail to clean table !!!!!!"); | ||||
| 				ex.printStackTrace(); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -0,0 +1,41 @@ | ||||
| package org.atriasoft.karideo.migration; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.dataAccess.DBAccess; | ||||
| import org.atriasoft.archidata.migration.MigrationSqlStep; | ||||
| import org.atriasoft.karideo.migration.model.TypeOld; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class Migration20250928 extends MigrationSqlStep { | ||||
| 	private static final Logger LOGGER = LoggerFactory.getLogger(Migration20250928.class); | ||||
|  | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
|  | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "migration-2025-10-12: Fix UserId in advencement"; | ||||
| 	} | ||||
|  | ||||
| 	public static ObjectId getElementType(final List<TypeOld> datas, final Long id) throws Exception { | ||||
| 		if (id == null) { | ||||
| 			return null; | ||||
| 		} | ||||
| 		for (final var elem : datas) { | ||||
| 			if (elem.id.equals(id)) { | ||||
| 				return elem.getOid(); | ||||
| 			} | ||||
| 		} | ||||
| 		throw new Exception("Does not exist"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		addAction((final DBAccess daMongo) -> { | ||||
|  | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										85
									
								
								back/src/org/atriasoft/karideo/migration/model/MediaOld.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								back/src/org/atriasoft/karideo/migration/model/MediaOld.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| package org.atriasoft.karideo.migration.model; | ||||
|  | ||||
| import java.util.Date; | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataJson; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.GenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
| import jakarta.validation.constraints.PositiveOrZero; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Entity | ||||
| @Table(name = "media") | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class MediaOld extends GenericDataSoftDelete { | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	public String name; | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Column(length = 0) | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Schema(description = "Foreign Key Id of the data") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Data.class) | ||||
| 	@Column(nullable = false) | ||||
| 	@CheckForeignKey(target = Data.class) | ||||
| 	public ObjectId dataId; | ||||
| 	@Schema(description = "Type of the media") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = TypeOld.class) | ||||
| 	@CheckForeignKey(target = TypeOld.class) | ||||
| 	public Long typeId; | ||||
| 	@Schema(description = "Series reference of the media") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = SeriesOld.class) | ||||
| 	@CheckForeignKey(target = SeriesOld.class) | ||||
| 	public Long seriesId; | ||||
| 	@Schema(description = "Season reference of the media") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = SeasonOld.class) | ||||
| 	@CheckForeignKey(target = SeasonOld.class) | ||||
| 	public Long seasonId; | ||||
| 	@Schema(description = "Episode Id") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer episode; | ||||
| 	@Schema(description = "Creation years of the media") | ||||
| 	public Date datePublication; | ||||
| 	@Schema(description = "Limitation Age of the media") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer ageLimit; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) ObjectId> covers = null; | ||||
|  | ||||
| 	@Override | ||||
| 	public String toString() { | ||||
| 		return "Media [name=" + this.name + ", description=" + this.description + ", dataId=" + this.dataId + ", typeId=" + this.typeId + ", seriesId=" + this.seriesId + ", seasonId=" + this.seasonId | ||||
| 				+ ", episode=" + this.episode + ", date=" + this.datePublication + ", ageLimit=" + this.ageLimit + ", covers=" + this.covers + "]"; | ||||
| 	} | ||||
|  | ||||
| 	@Column(nullable = true) | ||||
| 	private final ObjectId oid = new ObjectId(); | ||||
|  | ||||
| 	public ObjectId getOid() { | ||||
| 		return this.oid; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,57 @@ | ||||
| package org.atriasoft.karideo.migration.model; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.DataJson; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.GenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Table(name = "season") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class SeasonOld extends GenericDataSoftDelete { | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "series parent ID") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = SeriesOld.class) | ||||
| 	@CheckForeignKey(target = SeriesOld.class) | ||||
| 	public Long parentId; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) @NotNull ObjectId> covers = null; | ||||
|  | ||||
| 	@Column(nullable = true) | ||||
| 	private final ObjectId oid = new ObjectId(); | ||||
|  | ||||
| 	public ObjectId getOid() { | ||||
| 		return this.oid; | ||||
| 	} | ||||
| } | ||||
| @@ -0,0 +1,56 @@ | ||||
| package org.atriasoft.karideo.migration.model; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.DataJson; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.GenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Table(name = "series") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class SeriesOld extends GenericDataSoftDelete { | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "series parent ID") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = TypeOld.class) | ||||
| 	@CheckForeignKey(target = TypeOld.class) | ||||
| 	public Long parentId; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) @NotNull ObjectId> covers = null; | ||||
| 	@Column(nullable = true) | ||||
| 	private final ObjectId oid = new ObjectId(); | ||||
|  | ||||
| 	public ObjectId getOid() { | ||||
| 		return this.oid; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										58
									
								
								back/src/org/atriasoft/karideo/migration/model/TypeOld.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								back/src/org/atriasoft/karideo/migration/model/TypeOld.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| package org.atriasoft.karideo.migration.model; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.DataJson; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.GenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Table; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Table(name = "type") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class TypeOld extends GenericDataSoftDelete { | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) @NotNull ObjectId> covers = null; | ||||
|  | ||||
| 	public TypeOld() {} | ||||
|  | ||||
| 	public TypeOld(final String name, final String description) { | ||||
| 		this.name = name; | ||||
| 		this.description = description; | ||||
| 	} | ||||
|  | ||||
| 	@Column(nullable = true) | ||||
| 	private final ObjectId oid = new ObjectId(); | ||||
|  | ||||
| 	public ObjectId getOid() { | ||||
| 		return this.oid; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,57 @@ | ||||
| package org.atriasoft.karideo.migration.model; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.DataNotRead; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiAccessLimitation; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.model.GenericDataSoftDelete; | ||||
| import org.atriasoft.karideo.model.UserKarideo; | ||||
| import org.bson.types.ObjectId; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
| import jakarta.validation.constraints.PositiveOrZero; | ||||
|  | ||||
| @Table(name = "userMediaAdvancement") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class UserMediaAdvancementOld extends GenericDataSoftDelete { | ||||
| 	@DataNotRead | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Foreign Key Id of the user") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = UserKarideo.class) | ||||
| 	@Nullable | ||||
| 	@ApiAccessLimitation(updatable = false, creatable = false) | ||||
| 	public Long userId; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Id of the media") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = MediaOld.class) | ||||
| 	@CheckForeignKey(target = MediaOld.class) | ||||
| 	public Long mediaId; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Percent of advancement in the media") | ||||
| 	@PositiveOrZero | ||||
| 	public Float percent; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Number of second of advancement in the media") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer time; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Number of time this media has been read") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer count; | ||||
| 	@Column(nullable = true) | ||||
| 	private final ObjectId oid = new ObjectId(); | ||||
|  | ||||
| 	public ObjectId getOid() { | ||||
| 		return this.oid; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										68
									
								
								back/src/org/atriasoft/karideo/model/Media.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								back/src/org/atriasoft/karideo/model/Media.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| package org.atriasoft.karideo.model; | ||||
|  | ||||
| import java.util.Date; | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.OIDGenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.PositiveOrZero; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Entity() | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class Media extends OIDGenericDataSoftDelete { | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	@NotNull | ||||
| 	public String name; | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Schema(description = "Foreign Key Id of the data") | ||||
| 	@Column(nullable = false) | ||||
| 	@CheckForeignKey(target = Data.class) | ||||
| 	public ObjectId dataId; | ||||
| 	@Schema(description = "Type of the media") | ||||
| 	@CheckForeignKey(target = Type.class) | ||||
| 	public ObjectId typeId; | ||||
| 	@Schema(description = "Series reference of the media") | ||||
| 	@CheckForeignKey(target = Series.class) | ||||
| 	public ObjectId seriesId; | ||||
| 	@Schema(description = "Season reference of the media") | ||||
| 	@CheckForeignKey(target = Season.class) | ||||
| 	public ObjectId seasonId; | ||||
| 	@Schema(description = "Episode Id") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer episode; | ||||
| 	@Schema(description = "Creation years of the media") | ||||
| 	public Date datePublication; | ||||
| 	@Schema(description = "Limitation Age of the media") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer ageLimit; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) ObjectId> covers = null; | ||||
|  | ||||
| 	@Override | ||||
| 	public String toString() { | ||||
| 		return "Media [name=" + this.name + ", description=" + this.description + ", dataId=" + this.dataId + ", typeId=" + this.typeId + ", seriesId=" + this.seriesId + ", seasonId=" + this.seasonId | ||||
| 				+ ", episode=" + this.episode + ", date=" + this.datePublication + ", ageLimit=" + this.ageLimit + ", covers=" + this.covers + "]"; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										42
									
								
								back/src/org/atriasoft/karideo/model/Season.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								back/src/org/atriasoft/karideo/model/Season.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| package org.atriasoft.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.OIDGenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Entity() | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class Season extends OIDGenericDataSoftDelete { | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	public String name; | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Schema(description = "series parent ID") | ||||
| 	// @ManyToOneDoc(targetEntity = Series.class, remoteField = "") | ||||
| 	@CheckForeignKey(target = Series.class) | ||||
| 	public ObjectId seriesId; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) @NotNull ObjectId> covers = null; | ||||
| } | ||||
							
								
								
									
										45
									
								
								back/src/org/atriasoft/karideo/model/Series.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								back/src/org/atriasoft/karideo/model/Series.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| package org.atriasoft.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.OIDGenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Entity() | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class Series extends OIDGenericDataSoftDelete { | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	@NotNull | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "series parent ID") | ||||
| 	@CheckForeignKey(target = Type.class) | ||||
| 	public ObjectId typeId; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) @NotNull ObjectId> covers = null; | ||||
| } | ||||
							
								
								
									
										49
									
								
								back/src/org/atriasoft/karideo/model/Type.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								back/src/org/atriasoft/karideo/model/Type.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| package org.atriasoft.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.annotation.checker.CollectionNotEmpty; | ||||
| import org.atriasoft.archidata.model.Data; | ||||
| import org.atriasoft.archidata.model.OIDGenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
| import org.hibernate.validator.constraints.UniqueElements; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.annotation.Nullable; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.Size; | ||||
|  | ||||
| @Entity() | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class Type extends OIDGenericDataSoftDelete { | ||||
| 	@NotNull | ||||
| 	@Size(min = 0, max = 256) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	@Size(min = 0, max = 8192) | ||||
| 	public String description; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@Nullable | ||||
| 	@CollectionNotEmpty | ||||
| 	@UniqueElements | ||||
| 	public List<@CheckForeignKey(target = Data.class) @NotNull ObjectId> covers = null; | ||||
|  | ||||
| 	public Type() {} | ||||
|  | ||||
| 	public Type(final String name, final String description) { | ||||
| 		this.name = name; | ||||
| 		this.description = description; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package org.kar.karideo.model; | ||||
| package org.atriasoft.karideo.model; | ||||
| 
 | ||||
| import org.kar.archidata.annotation.DataIfNotExists; | ||||
| import org.kar.archidata.model.User; | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.model.User; | ||||
| 
 | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
| 
 | ||||
| @@ -0,0 +1,43 @@ | ||||
| package org.atriasoft.karideo.model; | ||||
|  | ||||
| import org.atriasoft.archidata.annotation.DataIfNotExists; | ||||
| import org.atriasoft.archidata.annotation.DataNotRead; | ||||
| import org.atriasoft.archidata.annotation.apiGenerator.ApiGenerationMode; | ||||
| import org.atriasoft.archidata.annotation.checker.CheckForeignKey; | ||||
| import org.atriasoft.archidata.model.OIDGenericDataSoftDelete; | ||||
| import org.bson.types.ObjectId; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.validation.constraints.NotNull; | ||||
| import jakarta.validation.constraints.PositiveOrZero; | ||||
|  | ||||
| @Entity() | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| @ApiGenerationMode(create = true, update = true) | ||||
| public class UserMediaAdvancement extends OIDGenericDataSoftDelete { | ||||
| 	@DataNotRead | ||||
| 	@NotNull | ||||
| 	@Schema(description = "Foreign Key Id of the user") | ||||
| 	@CheckForeignKey(target = UserKarideo.class) | ||||
| 	public ObjectId userId; | ||||
| 	@NotNull | ||||
| 	@Schema(description = "Id of the media") | ||||
| 	@CheckForeignKey(target = Media.class) | ||||
| 	public ObjectId mediaId; | ||||
| 	@NotNull | ||||
| 	@Schema(description = "Percent of advancement in the media") | ||||
| 	@PositiveOrZero | ||||
| 	public Float percent; | ||||
| 	@NotNull | ||||
| 	@Schema(description = "Number of second of advancement in the media") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer time; | ||||
| 	@NotNull | ||||
| 	@Schema(description = "Number of time this media has been read") | ||||
| 	@PositiveOrZero | ||||
| 	public Integer count; | ||||
| } | ||||
							
								
								
									
										29
									
								
								back/src/org/atriasoft/karideo/util/ConfigVariable.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								back/src/org/atriasoft/karideo/util/ConfigVariable.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| package org.atriasoft.karideo.util; | ||||
|  | ||||
| public class ConfigVariable { | ||||
| 	public static final String BASE_NAME = "ORG_KARIDEO_"; | ||||
|  | ||||
| 	public static String getFrontFolder() { | ||||
| 		final String out = System.getenv(BASE_NAME + "FRONT_FOLDER"); | ||||
| 		if (out == null) { | ||||
| 			return "./front"; | ||||
| 		} | ||||
| 		return out; | ||||
| 	} | ||||
|  | ||||
| 	public static String getBackupFolder() { | ||||
| 		final String out = System.getenv(BASE_NAME + "BACKUP_FOLDER"); | ||||
| 		if (out == null) { | ||||
| 			return "./backup"; | ||||
| 		} | ||||
| 		return out; | ||||
| 	} | ||||
|  | ||||
| 	public static boolean isInitWithBackup() { | ||||
| 		final String out = System.getenv(BASE_NAME + "INIT_WITH_BACKUP"); | ||||
| 		if (out == null) { | ||||
| 			return false; | ||||
| 		} | ||||
| 		return "true".equals(out); | ||||
| 	} | ||||
| } | ||||
| @@ -1,65 +0,0 @@ | ||||
|  | ||||
| package org.kar.karideo; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.kar.archidata.api.DataResource; | ||||
| import org.kar.archidata.dataAccess.DataFactoryTsApi; | ||||
| import org.kar.archidata.tools.ConfigBaseVariable; | ||||
| import org.kar.karideo.api.Front; | ||||
| import org.kar.karideo.api.HealthCheck; | ||||
| import org.kar.karideo.api.MediaResource; | ||||
| import org.kar.karideo.api.SeasonResource; | ||||
| import org.kar.karideo.api.SeriesResource; | ||||
| import org.kar.karideo.api.TypeResource; | ||||
| import org.kar.karideo.api.UserMediaAdvancementResource; | ||||
| import org.kar.karideo.api.UserResource; | ||||
| import org.kar.karideo.migration.Initialization; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class WebLauncherLocal extends WebLauncher { | ||||
| 	private final static Logger LOGGER = LoggerFactory.getLogger(WebLauncherLocal.class); | ||||
|  | ||||
| 	private WebLauncherLocal() {} | ||||
|  | ||||
| 	public static void main(final String[] args) throws Exception { | ||||
| 		DataFactoryTsApi.generatePackage(List.of( | ||||
| 				Front.class, | ||||
| 				HealthCheck.class, | ||||
| 				SeasonResource.class, | ||||
| 				SeriesResource.class, | ||||
| 				TypeResource.class, | ||||
| 				UserMediaAdvancementResource.class, | ||||
| 				UserResource.class, | ||||
| 				MediaResource.class, | ||||
| 				DataResource.class), | ||||
| 				Initialization.CLASSES_BASE, "../front/src/app/back-api/"); | ||||
| 		final WebLauncherLocal launcher = new WebLauncherLocal(); | ||||
| 		launcher.process(); | ||||
| 		LOGGER.info("end-configure the server & wait finish process:"); | ||||
| 		Thread.currentThread().join(); | ||||
| 		LOGGER.info("STOP the REST server:"); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void process() throws InterruptedException { | ||||
| 		if (true) { | ||||
| 			// for local test: | ||||
| 			ConfigBaseVariable.apiAdress = "http://0.0.0.0:18080/karideo/api/"; | ||||
| 			ConfigBaseVariable.dbPort = "3906"; | ||||
| 		} | ||||
| 		try { | ||||
| 			super.migrateDB(); | ||||
| 		} catch (final Exception e) { | ||||
| 			e.printStackTrace(); | ||||
| 			while (true) { | ||||
| 				LOGGER.error("============================================================================"); | ||||
| 				LOGGER.error("== Migration fail ==> waiting intervention of administrator..."); | ||||
| 				LOGGER.error("============================================================================"); | ||||
| 				Thread.sleep(60 * 60 * 1000); | ||||
| 			} | ||||
| 		} | ||||
| 		super.process(); | ||||
| 	} | ||||
| } | ||||
| @@ -1,123 +0,0 @@ | ||||
| package org.kar.karideo.api; | ||||
|  | ||||
| import java.io.InputStream; | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.glassfish.jersey.media.multipart.FormDataContentDisposition; | ||||
| import org.glassfish.jersey.media.multipart.FormDataParam; | ||||
| import org.kar.archidata.annotation.AsyncType; | ||||
| import org.kar.archidata.annotation.TypeScriptProgress; | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.dataAccess.QueryAnd; | ||||
| import org.kar.archidata.dataAccess.QueryCondition; | ||||
| import org.kar.archidata.dataAccess.addOn.AddOnDataJson; | ||||
| import org.kar.archidata.dataAccess.options.Condition; | ||||
| import org.kar.archidata.tools.DataTools; | ||||
| import org.kar.karideo.model.Season; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.PATCH; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
|  | ||||
| @Path("/season") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class SeasonResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(SeasonResource.class); | ||||
|  | ||||
|  | ||||
| 	@GET | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get a specific Season with his ID", tags = "GLOBAL") | ||||
| 	public List<Season> gets() throws Exception { | ||||
| 		return DataAccess.gets(Season.class); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Get all season", tags = "GLOBAL") | ||||
| 	public Season get(@PathParam("id") final Long id) throws Exception { | ||||
| 		return DataAccess.get(Season.class, id); | ||||
| 	} | ||||
|  | ||||
| 	/* ============================================================================= | ||||
| 	 * ADMIN SECTION: | ||||
| 	 * ============================================================================= */ | ||||
|  | ||||
| 	@POST | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Create a new season", tags = "GLOBAL") | ||||
| 	public Season post(final Season jsonRequest) throws Exception { | ||||
| 		return DataAccess.insert(jsonRequest); | ||||
| 	} | ||||
|  | ||||
| 	@PATCH | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific season", tags = "GLOBAL") | ||||
| 	public Season patch(@PathParam("id") final Long id, @AsyncType(Season.class) final String jsonRequest) throws Exception { | ||||
| 		DataAccess.updateWithJson(Season.class, id, jsonRequest); | ||||
| 		return DataAccess.get(Season.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific season", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("id") final Long id) throws Exception { | ||||
| 		DataAccess.delete(Season.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@POST | ||||
| 	@Path("{id}/cover") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.MULTIPART_FORM_DATA) | ||||
| 	@Operation(description = "Upload a new season cover season", tags = "GLOBAL") | ||||
| 	@TypeScriptProgress | ||||
| 	public Season uploadCover(@PathParam("id") final Long id, @FormDataParam("fileName") final String fileName, @FormDataParam("file") final InputStream fileInputStream, | ||||
| 			@FormDataParam("file") final FormDataContentDisposition fileMetaData) throws Exception { | ||||
| 		DataTools.uploadCover(Season.class, id, fileName, fileInputStream, fileMetaData); | ||||
| 		return DataAccess.get(Season.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{id}/cover/{coverId}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific cover of a season", tags = "GLOBAL") | ||||
| 	public Season removeCover(@PathParam("id") final Long id, @PathParam("coverId") final UUID coverId) throws Exception { | ||||
| 		AddOnDataJson.removeLink(Season.class, id, "covers", coverId); | ||||
| 		return DataAccess.get(Season.class, id); | ||||
| 	} | ||||
|  | ||||
| 	public static Season getOrCreate(final String name, final Long seriesId) { | ||||
| 		try { | ||||
| 			Season out = DataAccess.getWhere(Season.class, new Condition(new QueryAnd(new QueryCondition("name", "=", name), new QueryCondition("parentId", "=", seriesId)))); | ||||
| 			if (out == null) { | ||||
| 				out = new Season(); | ||||
| 				out.name = name; | ||||
| 				out.parentId = seriesId; | ||||
| 				out = DataAccess.insert(out); | ||||
| 			} | ||||
| 			return out; | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,122 +0,0 @@ | ||||
| package org.kar.karideo.api; | ||||
|  | ||||
| import java.io.InputStream; | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.glassfish.jersey.media.multipart.FormDataContentDisposition; | ||||
| import org.glassfish.jersey.media.multipart.FormDataParam; | ||||
| import org.kar.archidata.annotation.AsyncType; | ||||
| import org.kar.archidata.annotation.TypeScriptProgress; | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.dataAccess.QueryAnd; | ||||
| import org.kar.archidata.dataAccess.QueryCondition; | ||||
| import org.kar.archidata.dataAccess.addOn.AddOnDataJson; | ||||
| import org.kar.archidata.dataAccess.options.Condition; | ||||
| import org.kar.archidata.tools.DataTools; | ||||
| import org.kar.karideo.model.Series; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.PATCH; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
|  | ||||
| @Path("/series") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class SeriesResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(SeriesResource.class); | ||||
|  | ||||
| 	@GET | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get all Series", tags = "GLOBAL") | ||||
| 	public List<Series> gets() throws Exception { | ||||
| 		return DataAccess.gets(Series.class); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Get a specific Series with his ID", tags = "GLOBAL") | ||||
| 	public Series get(@PathParam("id") final Long id) throws Exception { | ||||
| 		return DataAccess.get(Series.class, id); | ||||
| 	} | ||||
|  | ||||
| 	/* ============================================================================= | ||||
| 	 * ADMIN SECTION: | ||||
| 	 * ============================================================================= */ | ||||
|  | ||||
| 	@POST | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Create a new Series", tags = "GLOBAL") | ||||
| 	public Series post(final Series jsonRequest) throws Exception { | ||||
| 		return DataAccess.insert(jsonRequest); | ||||
| 	} | ||||
|  | ||||
| 	@PATCH | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific Series", tags = "GLOBAL") | ||||
| 	public Series patch(@PathParam("id") final Long id, @AsyncType(Series.class) final String jsonRequest) throws Exception { | ||||
| 		DataAccess.updateWithJson(Series.class, id, jsonRequest); | ||||
| 		return DataAccess.get(Series.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific Series", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("id") final Long id) throws Exception { | ||||
| 		DataAccess.delete(Series.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@POST | ||||
| 	@Path("{id}/cover") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes({ MediaType.MULTIPART_FORM_DATA }) | ||||
| 	@Operation(description = "Upload a new season cover Series", tags = "GLOBAL") | ||||
| 	@TypeScriptProgress | ||||
| 	public Series uploadCover(@PathParam("id") final Long id, @FormDataParam("fileName") final String fileName, @FormDataParam("file") final InputStream fileInputStream, | ||||
| 			@FormDataParam("file") final FormDataContentDisposition fileMetaData) throws Exception { | ||||
| 		DataTools.uploadCover(Series.class, id, fileName, fileInputStream, fileMetaData); | ||||
| 		return DataAccess.get(Series.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{id}/cover/{coverId}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific Series of a season", tags = "GLOBAL") | ||||
| 	public Series removeCover(@PathParam("id") final Long id, @PathParam("coverId") final UUID coverId) throws Exception { | ||||
| 		AddOnDataJson.removeLink(Series.class, id, "covers", coverId); | ||||
| 		return DataAccess.get(Series.class, id); | ||||
| 	} | ||||
|  | ||||
| 	public static Series getOrCreate(final String name, final Long typeId) { | ||||
| 		try { | ||||
| 			Series out = DataAccess.getWhere(Series.class, new Condition(new QueryAnd(new QueryCondition("name", "=", name), new QueryCondition("parentId", "=", typeId)))); | ||||
| 			if (out == null) { | ||||
| 				out = new Series(); | ||||
| 				out.name = name; | ||||
| 				out.parentId = typeId; | ||||
| 				out = DataAccess.insert(out); | ||||
| 			} | ||||
| 			return out; | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,125 +0,0 @@ | ||||
| package org.kar.karideo.api; | ||||
|  | ||||
| import java.io.InputStream; | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.glassfish.jersey.media.multipart.FormDataContentDisposition; | ||||
| import org.glassfish.jersey.media.multipart.FormDataParam; | ||||
| import org.kar.archidata.annotation.AsyncType; | ||||
| import org.kar.archidata.annotation.TypeScriptProgress; | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.dataAccess.QueryCondition; | ||||
| import org.kar.archidata.dataAccess.addOn.AddOnDataJson; | ||||
| import org.kar.archidata.dataAccess.options.Condition; | ||||
| import org.kar.archidata.tools.DataTools; | ||||
| import org.kar.karideo.model.Type; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.Operation; | ||||
| import jakarta.annotation.security.RolesAllowed; | ||||
| import jakarta.ws.rs.Consumes; | ||||
| import jakarta.ws.rs.DELETE; | ||||
| import jakarta.ws.rs.GET; | ||||
| import jakarta.ws.rs.PATCH; | ||||
| import jakarta.ws.rs.POST; | ||||
| import jakarta.ws.rs.Path; | ||||
| import jakarta.ws.rs.PathParam; | ||||
| import jakarta.ws.rs.Produces; | ||||
| import jakarta.ws.rs.core.MediaType; | ||||
|  | ||||
| @Path("/type") | ||||
| @Produces(MediaType.APPLICATION_JSON) | ||||
| public class TypeResource { | ||||
| 	static final Logger LOGGER = LoggerFactory.getLogger(TypeResource.class); | ||||
|  | ||||
|  | ||||
| 	@GET | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Operation(description = "Get all Type", tags = "GLOBAL") | ||||
| 	public List<Type> gets() throws Exception { | ||||
| 		return DataAccess.gets(Type.class); | ||||
| 	} | ||||
|  | ||||
| 	@GET | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("USER") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Get a specific Type with his ID", tags = "GLOBAL") | ||||
| 	public Type get(@PathParam("id") final Long id) throws Exception { | ||||
| 		return DataAccess.get(Type.class, id); | ||||
| 	} | ||||
|  | ||||
| 	public static Type getId(final Long id) throws Exception { | ||||
| 		return DataAccess.get(Type.class, id); | ||||
| 	} | ||||
|  | ||||
| 	/* ============================================================================= | ||||
| 	 * ADMIN SECTION: | ||||
| 	 * ============================================================================= */ | ||||
|  | ||||
| 	@POST | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Create a new Type", tags = "GLOBAL") | ||||
| 	public Type post(final Type jsonRequest) throws Exception { | ||||
| 		return DataAccess.insert(jsonRequest); | ||||
| 	} | ||||
|  | ||||
| 	@PATCH | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes(MediaType.APPLICATION_JSON) | ||||
| 	@Operation(description = "Modify a specific Type", tags = "GLOBAL") | ||||
| 	public Type patch(@PathParam("id") final Long id, @AsyncType(Type.class) final String jsonRequest) throws Exception { | ||||
| 		DataAccess.updateWithJson(Type.class, id, jsonRequest); | ||||
| 		return DataAccess.get(Type.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{id}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific Type", tags = "GLOBAL") | ||||
| 	public void remove(@PathParam("id") final Long id) throws Exception { | ||||
| 		DataAccess.delete(Type.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@POST | ||||
| 	@Path("{id}/cover") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Consumes({ MediaType.MULTIPART_FORM_DATA }) | ||||
| 	@Operation(description = "Upload a new season cover Type", tags = "GLOBAL") | ||||
| 	@TypeScriptProgress | ||||
| 	public Type uploadCover(@PathParam("id") final Long id, @FormDataParam("fileName") final String fileName, @FormDataParam("file") final InputStream fileInputStream, | ||||
| 			@FormDataParam("file") final FormDataContentDisposition fileMetaData) throws Exception { | ||||
| 		DataTools.uploadCover(Type.class, id, fileName, fileInputStream, fileMetaData); | ||||
| 		return DataAccess.get(Type.class, id); | ||||
| 	} | ||||
|  | ||||
| 	@DELETE | ||||
| 	@Path("{id}/cover/{coverId}") | ||||
| 	@RolesAllowed("ADMIN") | ||||
| 	@Operation(description = "Remove a specific cover of a type", tags = "GLOBAL") | ||||
| 	public Type removeCover(@PathParam("id") final Long id, @PathParam("coverId") final UUID coverId) throws Exception { | ||||
| 		AddOnDataJson.removeLink(Type.class, id, "covers", coverId); | ||||
| 		return DataAccess.get(Type.class, id); | ||||
| 	} | ||||
|  | ||||
| 	public static Type getOrCreate(final String name) { | ||||
| 		try { | ||||
| 			Type out = DataAccess.getWhere(Type.class, new Condition(new QueryCondition("name", "=", name))); | ||||
| 			if (out == null) { | ||||
| 				out = new Type(); | ||||
| 				out.name = name; | ||||
| 				out = DataAccess.insert(out); | ||||
| 			} | ||||
| 			return out; | ||||
| 		} catch (final Exception e) { | ||||
| 			// TODO Auto-generated catch block | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,60 +0,0 @@ | ||||
| package org.kar.karideo.internal; | ||||
|  | ||||
| //import io.scenarium.logger.LogLevel; | ||||
| //import io.scenarium.logger.Logger; | ||||
|  | ||||
| public class Log { | ||||
| //	private static final String LIB_NAME = "logger"; | ||||
| //	private static final String LIB_NAME_DRAW = Logger.getDrawableName(LIB_NAME); | ||||
| //	private static final boolean PRINT_CRITICAL = Logger.getNeedPrint(LIB_NAME, LogLevel.CRITICAL); | ||||
| //	private static final boolean PRINT_ERROR = Logger.getNeedPrint(LIB_NAME, LogLevel.ERROR); | ||||
| //	private static final boolean PRINT_WARNING = Logger.getNeedPrint(LIB_NAME, LogLevel.WARNING); | ||||
| //	private static final boolean PRINT_INFO = Logger.getNeedPrint(LIB_NAME, LogLevel.INFO); | ||||
| //	private static final boolean PRINT_DEBUG = Logger.getNeedPrint(LIB_NAME, LogLevel.DEBUG); | ||||
| //	private static final boolean PRINT_VERBOSE = Logger.getNeedPrint(LIB_NAME, LogLevel.VERBOSE); | ||||
| //	private static final boolean PRINT_TODO = Logger.getNeedPrint(LIB_NAME, LogLevel.TODO); | ||||
| //	private static final boolean PRINT_PRINT = Logger.getNeedPrint(LIB_NAME, LogLevel.PRINT); | ||||
| // | ||||
| //	private Log() {} | ||||
| // | ||||
| //	public static void print(String data) { | ||||
| //		if (PRINT_PRINT) | ||||
| //			Logger.print(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void todo(String data) { | ||||
| //		if (PRINT_TODO) | ||||
| //			Logger.todo(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void critical(String data) { | ||||
| //		if (PRINT_CRITICAL) | ||||
| //			Logger.critical(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void error(String data) { | ||||
| //		if (PRINT_ERROR) | ||||
| //			Logger.error(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void warning(String data) { | ||||
| //		if (PRINT_WARNING) | ||||
| //			Logger.warning(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void info(String data) { | ||||
| //		if (PRINT_INFO) | ||||
| //			Logger.info(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void debug(String data) { | ||||
| //		if (PRINT_DEBUG) | ||||
| //			Logger.debug(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
| // | ||||
| //	public static void verbose(String data) { | ||||
| //		if (PRINT_VERBOSE) | ||||
| //			Logger.verbose(LIB_NAME_DRAW, data); | ||||
| //	} | ||||
|  | ||||
| } | ||||
| @@ -1,69 +0,0 @@ | ||||
| package org.kar.karideo.migration; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.kar.archidata.migration.MigrationSqlStep; | ||||
| import org.kar.archidata.model.Data; | ||||
| import org.kar.archidata.model.User; | ||||
| import org.kar.karideo.model.Media; | ||||
| import org.kar.karideo.model.Season; | ||||
| import org.kar.karideo.model.Series; | ||||
| import org.kar.karideo.model.Type; | ||||
| import org.kar.karideo.model.UserMediaAdvancement; | ||||
|  | ||||
| public class Initialization extends MigrationSqlStep { | ||||
|  | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
|  | ||||
| 	public static final List<Class<?>> CLASSES_BASE = List.of(Data.class, Media.class, Type.class, Series.class, Season.class, User.class, UserMediaAdvancement.class); | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "Initialization"; | ||||
| 	} | ||||
|  | ||||
| 	public Initialization() { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		addClass(Data.class); | ||||
| 		addClass(Media.class); | ||||
| 		addClass(Type.class); | ||||
| 		addClass(Series.class); | ||||
| 		addClass(Season.class); | ||||
| 		addClass(User.class); | ||||
| 		addClass(UserMediaAdvancement.class); | ||||
|  | ||||
| 		addAction(""" | ||||
| 				INSERT INTO `type` (`id`, `name`, `description`) VALUES | ||||
| 				(UUID_TO_BIN('15237fd7-d4ee-11ee-a8dd-02420a030203'), 'Documentary',       'Documentary (animals, space, earth...)'), | ||||
| 				(UUID_TO_BIN('553146c1-d4ee-11ee-a8dd-02420a030203'), 'Movie',             'Movie with real humans (film)'), | ||||
| 				(UUID_TO_BIN('59c430a3-d4ee-11ee-a8dd-02420a030203'), 'Animation',         'Animation movies (film)'), | ||||
| 				(UUID_TO_BIN('5cd619e3-d4ee-11ee-a8dd-02420a030203'), 'Short movie',       'Small movies (less 2 minutes)'), | ||||
| 				(UUID_TO_BIN('5fbbf085-d4ee-11ee-a8dd-02420a030203'), 'TV show',           'TV show for old peoples'), | ||||
| 				(UUID_TO_BIN('66dcb6ba-d4ee-11ee-a8dd-02420a030203'), 'Animation TV show', 'TV show for young peoples'), | ||||
| 				(UUID_TO_BIN('69ee5c15-d4ee-11ee-a8dd-02420a030203'), 'Theater',           'Theater play'), | ||||
| 				(UUID_TO_BIN('6ce72530-d4ee-11ee-a8dd-02420a030203'), 'One man show',      'Recorded stand up'), | ||||
| 				(UUID_TO_BIN('6ff1691a-d4ee-11ee-a8dd-02420a030203'), 'Concert',           'Recorded concert'), | ||||
| 				(UUID_TO_BIN('730815ef-d4ee-11ee-a8dd-02420a030203'), 'Opera',             'Recorded opera'); | ||||
| 				"""); | ||||
| 		// set start increment element to permit to add after default elements | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `media` AUTO_INCREMENT = 1000; | ||||
| 				""", "mysql"); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `type` AUTO_INCREMENT = 1000; | ||||
| 				""", "mysql"); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `series` AUTO_INCREMENT = 1000; | ||||
| 				""", "mysql"); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `season` AUTO_INCREMENT = 1000; | ||||
| 				""", "mysql"); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `userMediaAdvancement` AUTO_INCREMENT = 1000; | ||||
| 				""", "mysql"); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,28 +0,0 @@ | ||||
| package org.kar.karideo.migration; | ||||
|  | ||||
| import org.kar.archidata.migration.MigrationSqlStep; | ||||
| import org.kar.karideo.model.UserMediaAdvancement; | ||||
|  | ||||
| public class Migration20230810 extends MigrationSqlStep { | ||||
| 	 | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
| 	 | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "migration-2023-08-10"; | ||||
| 	} | ||||
| 	 | ||||
| 	public Migration20230810() { | ||||
| 		 | ||||
| 	} | ||||
| 	 | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		addClass(UserMediaAdvancement.class); | ||||
| 		 | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `userMediaAdvancement` AUTO_INCREMENT = 1000; | ||||
| 				"""); | ||||
| 	} | ||||
| 	 | ||||
| } | ||||
| @@ -1,32 +0,0 @@ | ||||
| package org.kar.karideo.migration; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import org.kar.archidata.migration.MigrationSqlStep; | ||||
|  | ||||
| public class Migration20231015 extends MigrationSqlStep { | ||||
|  | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
|  | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "migration-2023-10-15: refactor creation and update time"; | ||||
| 	} | ||||
|  | ||||
| 	public Migration20231015() { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		for (String elem : List.of("data", "media", "media_link_cover", "season", "season_link_cover", "series", "series_link_cover", "type", "type_link_cover", "user", "userMediaAdvancement")) { | ||||
| 			addAction(""" | ||||
| 					ALTER TABLE `""" + elem + """ | ||||
| 					` | ||||
| 						RENAME COLUMN `create_date` TO `createdAt`, | ||||
| 						RENAME COLUMN `modify_date` TO `updatedAt`; | ||||
| 					"""); | ||||
| 		} | ||||
| 		display(); | ||||
| 	} | ||||
| } | ||||
| @@ -1,160 +0,0 @@ | ||||
| package org.kar.karideo.migration; | ||||
|  | ||||
| import org.kar.archidata.migration.MigrationSqlStep; | ||||
|  | ||||
| public class Migration20231126 extends MigrationSqlStep { | ||||
|  | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
|  | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "migration-2023-11-26: re-order the migration for the new API of archidata"; | ||||
| 	} | ||||
|  | ||||
| 	public Migration20231126() { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		 | ||||
| 		// update migration update (last one) | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `KAR_migration` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `create_date` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `modify_date` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				ADD `version` int NOT NULL DEFAULT '2' AFTER `deleted`, | ||||
| 				CHANGE `name` `name` varchar(256) COLLATE 'utf8mb4_0900_ai_ci' NULL COMMENT 'Name of the migration' AFTER `version`, | ||||
| 				CHANGE `terminated` `terminated` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'if the migration is well terminated or not' AFTER `name`, | ||||
| 				CHANGE `stepId` `stepId` int NULL COMMENT 'index in the migration progression' AFTER `terminated`, | ||||
| 				CHANGE `count` `count` int NULL COMMENT 'number of element in the migration' AFTER `stepId`, | ||||
| 				CHANGE `log` `log` text COLLATE 'utf8mb3_general_ci' NULL COMMENT 'Log generate by the migration' AFTER `count`; | ||||
| 				"""); | ||||
|  | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				CHANGE `sha512` `sha512` varchar(128) COLLATE 'utf8mb4_0900_ai_ci' NOT NULL COMMENT 'Sha512 of the data' AFTER `deleted`, | ||||
| 				CHANGE `mimeType` `mimeType` varchar(128) COLLATE 'utf8mb4_0900_ai_ci' NOT NULL COMMENT 'Mime -type of the media' AFTER `sha512`, | ||||
| 				CHANGE `size` `size` bigint NOT NULL COMMENT 'Size in Byte of the data' AFTER `mimeType`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `media` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				CHANGE `name` `name` text COLLATE 'utf8mb3_general_ci' NOT NULL COMMENT 'Name of the media (this represent the title)' AFTER `deleted`, | ||||
| 				CHANGE `description` `description` text COLLATE 'utf8mb3_general_ci' NULL COMMENT 'Description of the media' AFTER `name`, | ||||
| 				CHANGE `dataId` `dataId` bigint NOT NULL COMMENT 'Foreign Key Id of the data' AFTER `description`, | ||||
| 				CHANGE `typeId` `typeId` bigint NULL COMMENT 'Type of the media' AFTER `dataId`, | ||||
| 				CHANGE `seriesId` `seriesId` bigint NULL COMMENT 'Series reference of the media' AFTER `typeId`, | ||||
| 				CHANGE `seasonId` `seasonId` bigint NULL COMMENT 'Saison reference of the media' AFTER `seriesId`, | ||||
| 				CHANGE `episode` `episode` int NULL COMMENT 'Episide Id' AFTER `seasonId`, | ||||
| 				CHANGE `date` `date` int NULL AFTER `episode`, | ||||
| 				CHANGE `time` `time` int NULL COMMENT 'Creation years of the media' AFTER `date`, | ||||
| 				CHANGE `ageLimit` `ageLimit` int NULL COMMENT 'Limitation Age of the media' AFTER `time`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `media_link_cover` | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' AFTER `updatedAt`, | ||||
| 				CHANGE `media_id` `object1id` bigint NOT NULL AFTER `deleted`, | ||||
| 				CHANGE `cover_id` `object2id` bigint NOT NULL AFTER `object1id`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `season` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				CHANGE `name` `name` text COLLATE 'utf8mb3_general_ci' NOT NULL COMMENT 'Name of the media (this represent the title)' AFTER `deleted`, | ||||
| 				CHANGE `description` `description` text COLLATE 'utf8mb3_general_ci' NULL COMMENT 'Description of the media' AFTER `name`, | ||||
| 				CHANGE `parentId` `parentId` bigint NOT NULL COMMENT 'series parent ID' AFTER `description`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `season_link_cover` | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' AFTER `updatedAt`, | ||||
| 				CHANGE `season_id` `object1id` bigint NOT NULL AFTER `deleted`, | ||||
| 				CHANGE `cover_id` `object2id` bigint NOT NULL AFTER `object1id`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `series` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				CHANGE `name` `name` text COLLATE 'utf8mb3_general_ci' NOT NULL COMMENT 'Name of the media (this represent the title)' AFTER `deleted`, | ||||
| 				CHANGE `description` `description` text COLLATE 'utf8mb3_general_ci' NULL COMMENT 'Description of the media' AFTER `name`, | ||||
| 				CHANGE `parentId` `parentId` bigint NOT NULL COMMENT 'series parent ID' AFTER `description`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `series_link_cover` | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' AFTER `updatedAt`, | ||||
| 				CHANGE `series_id` `object1id` bigint NOT NULL AFTER `deleted`, | ||||
| 				CHANGE `cover_id` `object2id` bigint NOT NULL AFTER `object1id`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `type` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `name` `name` text COLLATE 'utf8mb3_general_ci' NOT NULL COMMENT 'Name of the media (this represent the title)' AFTER `deleted`, | ||||
| 				CHANGE `description` `description` text COLLATE 'utf8mb3_general_ci' NULL COMMENT 'Description of the media' AFTER `name`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `type_link_cover` | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' AFTER `updatedAt`, | ||||
| 				CHANGE `type_id` `object1id` bigint NOT NULL AFTER `deleted`, | ||||
| 				CHANGE `cover_id` `object2id` bigint NOT NULL AFTER `object1id`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `user` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				CHANGE `login` `login` varchar(128) COLLATE 'utf8mb4_0900_ai_ci' NULL AFTER `deleted`, | ||||
| 				CHANGE `lastConnection` `lastConnection` timestamp(3) NULL AFTER `login`, | ||||
| 				CHANGE `admin` `admin` tinyint(1) NOT NULL DEFAULT '0' AFTER `lastConnection`, | ||||
| 				CHANGE `blocked` `blocked` tinyint(1) NOT NULL DEFAULT '0' AFTER `admin`, | ||||
| 				CHANGE `removed` `removed` tinyint(1) NOT NULL DEFAULT '0' AFTER `blocked`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `userMediaAdvancement` | ||||
| 				CHANGE `id` `id` bigint NOT NULL COMMENT 'Primary key of the base' AUTO_INCREMENT FIRST, | ||||
| 				CHANGE `createdAt` `createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' AFTER `id`, | ||||
| 				CHANGE `updatedAt` `updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'When update the object' AFTER `createdAt`, | ||||
| 				CHANGE `deleted` `deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' AFTER `updatedAt`, | ||||
| 				CHANGE `userId` `userId` bigint NOT NULL COMMENT 'Foreign Key Id of the user' AFTER `deleted`, | ||||
| 				CHANGE `mediaId` `mediaId` bigint NOT NULL COMMENT 'Id of the media' AFTER `userId`, | ||||
| 				CHANGE `percent` `percent` float NOT NULL COMMENT 'Percent of admencement in the media' AFTER `mediaId`, | ||||
| 				CHANGE `time` `time` int NOT NULL COMMENT 'Number of second of admencement in the media' AFTER `percent`, | ||||
| 				CHANGE `count` `count` int NOT NULL COMMENT 'Number of time this media has been read' AFTER `time`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				CREATE TABLE `user_link_cover` ( | ||||
| 					`id` bigint NOT NULL AUTO_INCREMENT COMMENT 'Primary key of the base' , | ||||
| 					`createdAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) COMMENT 'Create time of the object' , | ||||
| 					`updatedAt` timestamp(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) ON UPDATE CURRENT_TIMESTAMP(3) COMMENT 'When update the object' , | ||||
| 					`deleted` tinyint(1) NOT NULL DEFAULT '0' COMMENT 'When delete, they are not removed, they are just set in a deleted state' , | ||||
| 					`object1Id` bigint NOT NULL COMMENT 'Object reference 1' , | ||||
| 					`object2Id` bigint NOT NULL COMMENT 'Object reference 2' , | ||||
| 				PRIMARY KEY (`id`) | ||||
| 				) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci; | ||||
| 				"""); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,141 +0,0 @@ | ||||
| package org.kar.karideo.migration; | ||||
|  | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.NoSuchFileException; | ||||
| import java.nio.file.Paths; | ||||
| import java.nio.file.StandardCopyOption; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.kar.archidata.api.DataResource; | ||||
| import org.kar.archidata.dataAccess.DataAccess; | ||||
| import org.kar.archidata.dataAccess.addOn.model.LinkTable; | ||||
| import org.kar.archidata.dataAccess.options.AccessDeletedItems; | ||||
| import org.kar.archidata.dataAccess.options.OverrideTableName; | ||||
| import org.kar.archidata.migration.MigrationSqlStep; | ||||
| import org.kar.archidata.tools.UuidUtils; | ||||
| import org.kar.karideo.migration.model.CoverConversion; | ||||
| import org.kar.karideo.migration.model.MediaConversion; | ||||
| import org.kar.karideo.migration.model.UUIDConversion; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class Migration20240226 extends MigrationSqlStep { | ||||
| 	private static final Logger LOGGER = LoggerFactory.getLogger(Migration20240226.class); | ||||
|  | ||||
| 	public static final int KARSO_INITIALISATION_ID = 1; | ||||
|  | ||||
| 	@Override | ||||
| 	public String getName() { | ||||
| 		return "migration-2024-02-26: convert base with UUID"; | ||||
| 	} | ||||
|  | ||||
| 	public Migration20240226() { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void generateStep() throws Exception { | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` ADD `uuid` binary(16) AFTER `id`; | ||||
| 				"""); | ||||
| 		addAction(() -> { | ||||
| 			final List<UUIDConversion> datas = DataAccess.gets(UUIDConversion.class, new AccessDeletedItems(), new OverrideTableName("data")); | ||||
| 			for (final UUIDConversion elem: datas) { | ||||
| 				elem.uuid = UuidUtils.nextUUID(); | ||||
| 			} | ||||
| 			for (final UUIDConversion elem: datas) { | ||||
| 				DataAccess.update(elem, elem.id, List.of("uuid"), new OverrideTableName("data")); | ||||
| 			} | ||||
| 		}); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` CHANGE `uuid` `uuid` binary(16) DEFAULT (UUID_TO_BIN(UUID(), TRUE)); | ||||
| 				"""); | ||||
| 		final List<String> tableToTransform = List.of("media", "season", "series", "type", "user"); | ||||
| 		for (final String tableName : tableToTransform ) { | ||||
| 			addAction("ALTER TABLE `" + tableName + "` ADD `covers` text NULL;"); | ||||
| 			addAction(() -> { | ||||
| 				final List<UUIDConversion> datas = DataAccess.gets(UUIDConversion.class, new AccessDeletedItems(), new OverrideTableName("data")); | ||||
| 				final List<CoverConversion> medias = DataAccess.gets(CoverConversion.class, new AccessDeletedItems(), new OverrideTableName(tableName)); | ||||
| 				final List<LinkTable> links = DataAccess.gets(LinkTable.class, new OverrideTableName(tableName + "_link_cover")); | ||||
| 				LOGGER.info("Get somes data: {} {} {}", datas.size(), medias.size(), links.size()); | ||||
| 				for (final CoverConversion media: medias) { | ||||
| 					final List<UUID> values = new ArrayList<>(); | ||||
| 					for (final LinkTable link: links) { | ||||
| 						if (link.object1Id.equals(media.id)) { | ||||
| 							for (final UUIDConversion data: datas) { | ||||
| 								if (data.id.equals(link.object2Id)) { | ||||
| 									values.add(data.uuid); | ||||
| 									break; | ||||
| 								} | ||||
| 							} | ||||
| 							break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (values.size() != 0) { | ||||
| 						media.covers = values; | ||||
| 						LOGGER.info("    update: {} => {}", media.id, media.covers); | ||||
| 						DataAccess.update(media, media.id, List.of("covers"), new OverrideTableName(tableName)); | ||||
| 					} | ||||
| 				} | ||||
| 			}); | ||||
| 			addAction("DROP TABLE `" + tableName + "_link_cover`;"); | ||||
| 		} | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `media` ADD `dataUUID` binary(16) AFTER dataId; | ||||
| 				"""); | ||||
| 		addAction(() -> { | ||||
| 			final List<UUIDConversion> datas = DataAccess.gets(UUIDConversion.class, new AccessDeletedItems(), new OverrideTableName("data")); | ||||
| 			final List<MediaConversion> medias = DataAccess.gets(MediaConversion.class, new AccessDeletedItems(), new OverrideTableName("media")); | ||||
| 			for (final MediaConversion media: medias) { | ||||
| 				for (final UUIDConversion data: datas) { | ||||
| 					if (data.id.equals(media.dataId)) { | ||||
| 						media.dataUUID = data.uuid; | ||||
| 						DataAccess.update(media, media.id, List.of("dataUUID"), new OverrideTableName("media")); | ||||
| 						break; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `media` DROP `dataId`; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `media` CHANGE `dataUUID` `dataId` binary(16) NOT NULL; | ||||
| 				"""); | ||||
| 		// Move the files... | ||||
| 		addAction(() -> { | ||||
| 			final List<UUIDConversion> datas = DataAccess.gets(UUIDConversion.class, new AccessDeletedItems(), new OverrideTableName("data")); | ||||
| 			for (final UUIDConversion data: datas) { | ||||
| 				final String origin = DataResource.getFileDataOld(data.id); | ||||
| 				final String destination = DataResource.getFileData(data.uuid); | ||||
| 				LOGGER.info("move file = {}", origin); | ||||
| 				LOGGER.info("        ==> {}", destination); | ||||
| 				try { | ||||
| 					Files.move(Paths.get(origin), Paths.get(destination), StandardCopyOption.ATOMIC_MOVE); | ||||
| 				} catch (final NoSuchFileException ex) { | ||||
| 					LOGGER.error("MOVE_ERROR : {} -> {}", origin, destination); | ||||
| 				} | ||||
| 			} | ||||
| 		}); | ||||
| 		/* I am not sure then I prefer keep the primary key for the moment | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` DROP `id`; | ||||
| 				"""); | ||||
| 		 */ | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` CHANGE `id` `idOld` bigint NOT NULL DEFAULT 0; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` DROP PRIMARY KEY; | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` CHANGE `uuid` `id` binary(16) DEFAULT (UUID_TO_BIN(UUID(), TRUE)); | ||||
| 				"""); | ||||
| 		addAction(""" | ||||
| 				ALTER TABLE `data` ADD PRIMARY KEY `id` (`id`); | ||||
| 				"""); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,15 +0,0 @@ | ||||
| package org.kar.karideo.migration.model; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.kar.archidata.annotation.DataJson; | ||||
|  | ||||
| import jakarta.persistence.Id; | ||||
|  | ||||
| public class CoverConversion { | ||||
| 	@Id | ||||
| 	public Long id = null; | ||||
| 	@DataJson | ||||
| 	public List<UUID> covers = null; | ||||
| } | ||||
| @@ -1,12 +0,0 @@ | ||||
| package org.kar.karideo.migration.model; | ||||
|  | ||||
| import java.util.UUID; | ||||
|  | ||||
| import jakarta.persistence.Id; | ||||
|  | ||||
| public class MediaConversion { | ||||
| 	@Id | ||||
| 	public Long id = null; | ||||
| 	public Long dataId = null; | ||||
| 	public UUID dataUUID = null; | ||||
| } | ||||
| @@ -1,11 +0,0 @@ | ||||
| package org.kar.karideo.migration.model; | ||||
|  | ||||
| import java.util.UUID; | ||||
|  | ||||
| import jakarta.persistence.Id; | ||||
|  | ||||
| public class UUIDConversion { | ||||
| 	@Id | ||||
| 	public Long id = null; | ||||
| 	public UUID uuid = null; | ||||
| } | ||||
| @@ -1,61 +0,0 @@ | ||||
| package org.kar.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.kar.archidata.annotation.DataJson; | ||||
| import org.kar.archidata.model.Data; | ||||
| import org.kar.archidata.model.GenericDataSoftDelete; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Entity; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
|  | ||||
| @Entity | ||||
| @Table(name = "media") | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| public class Media extends GenericDataSoftDelete { | ||||
| 	// Name of the media (this represent the title) | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	public String name; | ||||
| 	// Description of the media | ||||
| 	@Column(length = 0) | ||||
| 	public String description; | ||||
| 	// Foreign Key Id of the data | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Data.class) | ||||
| 	@Column(nullable = false) | ||||
| 	public UUID dataId; | ||||
| 	// Type of the media") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Type.class) | ||||
| 	public Long typeId; | ||||
| 	// Series reference of the media | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Series.class) | ||||
| 	public Long seriesId; | ||||
| 	// Season reference of the media | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Season.class) | ||||
| 	public Long seasonId; | ||||
| 	// Episode Id | ||||
| 	public Integer episode; | ||||
| 	// ") | ||||
| 	public Integer date; | ||||
| 	// Creation years of the media | ||||
| 	public Integer time; | ||||
| 	// Limitation Age of the media | ||||
| 	public Integer ageLimit; | ||||
| 	// List of Id of the specific covers | ||||
| 	@DataJson(targetEntity = Data.class) | ||||
| 	public List<UUID> covers = null; | ||||
|  | ||||
| 	@Override | ||||
| 	public String toString() { | ||||
| 		return "Media [name=" + this.name + ", description=" + this.description + ", dataId=" + this.dataId + ", typeId=" + this.typeId | ||||
| 				+ ", seriesId=" + this.seriesId + ", seasonId=" + this.seasonId + ", episode=" + this.episode + ", date=" + this.date | ||||
| 				+ ", time=" + this.time + ", ageLimit=" + this.ageLimit + ", covers=" + this.covers + "]"; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| } | ||||
| @@ -1,35 +0,0 @@ | ||||
| package org.kar.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.kar.archidata.annotation.DataIfNotExists; | ||||
| import org.kar.archidata.annotation.DataJson; | ||||
| import org.kar.archidata.model.GenericDataSoftDelete; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
|  | ||||
| @Table(name = "season") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| public class Season extends GenericDataSoftDelete { | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	public String description; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "series parent ID") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Series.class) | ||||
| 	public Long parentId; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	public List<UUID> covers = null; | ||||
| } | ||||
| @@ -1,35 +0,0 @@ | ||||
| package org.kar.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.kar.archidata.annotation.DataIfNotExists; | ||||
| import org.kar.archidata.annotation.DataJson; | ||||
| import org.kar.archidata.model.GenericDataSoftDelete; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
|  | ||||
| @Table(name = "series") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| public class Series extends GenericDataSoftDelete { | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	public String description; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "series parent ID") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Type.class) | ||||
| 	public Long parentId; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	public List<UUID> covers = null; | ||||
| } | ||||
| @@ -1,29 +0,0 @@ | ||||
| package org.kar.karideo.model; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import org.kar.archidata.annotation.DataIfNotExists; | ||||
| import org.kar.archidata.annotation.DataJson; | ||||
| import org.kar.archidata.model.GenericDataSoftDelete; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.Table; | ||||
|  | ||||
| @Table(name = "type") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| public class Type extends GenericDataSoftDelete { | ||||
| 	@Column(nullable = false, length = 0) | ||||
| 	@Schema(description = "Name of the media (this represent the title)") | ||||
| 	public String name; | ||||
| 	@Column(length = 0) | ||||
| 	@Schema(description = "Description of the media") | ||||
| 	public String description; | ||||
| 	@Schema(description = "List of Id of the specific covers") | ||||
| 	@DataJson() | ||||
| 	public List<UUID> covers = null; | ||||
| } | ||||
| @@ -1,37 +0,0 @@ | ||||
| package org.kar.karideo.model; | ||||
|  | ||||
| import org.kar.archidata.annotation.DataIfNotExists; | ||||
| import org.kar.archidata.annotation.DataNotRead; | ||||
| import org.kar.archidata.model.GenericDataSoftDelete; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
|  | ||||
| import io.swagger.v3.oas.annotations.media.Schema; | ||||
| import jakarta.persistence.Column; | ||||
| import jakarta.persistence.FetchType; | ||||
| import jakarta.persistence.ManyToOne; | ||||
| import jakarta.persistence.Table; | ||||
|  | ||||
| @Table(name = "userMediaAdvancement") | ||||
| @DataIfNotExists | ||||
| @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
| public class UserMediaAdvancement extends GenericDataSoftDelete { | ||||
| 	@DataNotRead | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Foreign Key Id of the user") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = UserKarideo.class) | ||||
| 	public Long userId; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Id of the media") | ||||
| 	@ManyToOne(fetch = FetchType.LAZY, targetEntity = Media.class) | ||||
| 	public Long mediaId; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Percent of advancement in the media") | ||||
| 	public Float percent; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Number of second of advancement in the media") | ||||
| 	public Integer time; | ||||
| 	@Column(nullable = false) | ||||
| 	@Schema(description = "Number of time this media has been read") | ||||
| 	public Integer count; | ||||
| } | ||||
| @@ -1,13 +0,0 @@ | ||||
| package org.kar.karideo.util; | ||||
|  | ||||
| public class ConfigVariable { | ||||
| 	public static final String BASE_NAME = "ORG_KARIDEO_"; | ||||
| 	 | ||||
|     public static String getFrontFolder() { | ||||
|         String out = System.getenv(BASE_NAME + "FRONT_FOLDER"); | ||||
|         if (out == null) { | ||||
|             return "/application/front"; | ||||
|         } | ||||
|         return out; | ||||
|     } | ||||
| } | ||||
| @@ -1,10 +1,11 @@ | ||||
| package test.kar.karideo; | ||||
| package test.atriasoft.karideo; | ||||
| 
 | ||||
| import java.util.Map; | ||||
| 
 | ||||
| import org.kar.archidata.tools.JWTWrapper; | ||||
| import org.atriasoft.archidata.filter.PartRight; | ||||
| import org.atriasoft.archidata.tools.JWTWrapper; | ||||
| 
 | ||||
| public class Common { | ||||
| 	static String USER_TOKEN = JWTWrapper.createJwtTestToken(16512, "test_user_login", "KarAuth", "karideo", Map.of("karideo", Map.of("USER", Boolean.TRUE))); | ||||
| 	static String ADMIN_TOKEN = JWTWrapper.createJwtTestToken(16512, "test_admin_login", "KarAuth", "karideo", Map.of("karideo", Map.of("USER", Boolean.TRUE, "ADMIN", Boolean.TRUE))); | ||||
| 	static String USER_TOKEN = JWTWrapper.createJwtTestToken(16512, "test_user_login", "KarAuth", "karideo", Map.of("karideo", Map.of("USER", PartRight.READ))); | ||||
| 	static String ADMIN_TOKEN = JWTWrapper.createJwtTestToken(16512, "test_admin_login", "KarAuth", "karideo", Map.of("karideo", Map.of("USER", PartRight.READ_WRITE, "ADMIN", PartRight.READ_WRITE))); | ||||
| } | ||||
							
								
								
									
										115
									
								
								back/test/src/test/atriasoft/karideo/ConfigureDb.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								back/test/src/test/atriasoft/karideo/ConfigureDb.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,115 @@ | ||||
| package test.atriasoft.karideo; | ||||
|  | ||||
| import java.io.IOException; | ||||
|  | ||||
| import org.atriasoft.archidata.dataAccess.DBAccess; | ||||
| import org.atriasoft.archidata.db.DbConfig; | ||||
| import org.atriasoft.archidata.db.DbIoFactory; | ||||
| import org.atriasoft.archidata.exception.DataAccessException; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import jakarta.ws.rs.InternalServerErrorException; | ||||
|  | ||||
| public class ConfigureDb { | ||||
| 	final static private Logger LOGGER = LoggerFactory.getLogger(ConfigureDb.class); | ||||
| 	final static private String modeTestForced = null;// "MONGO"; | ||||
| 	public static DBAccess da = null; | ||||
|  | ||||
| 	public static void configure() throws IOException, InternalServerErrorException, DataAccessException { | ||||
| 		String modeTest = System.getenv("TEST_E2E_MODE"); | ||||
| 		if (modeTest == null || modeTest.isEmpty() || "false".equalsIgnoreCase(modeTest)) { | ||||
| 			modeTest = "SQLITE-MEMORY"; | ||||
| 		} else if ("true".equalsIgnoreCase(modeTest)) { | ||||
| 			modeTest = "MY-SQL"; | ||||
| 		} | ||||
| 		// override the local test: | ||||
| 		if (modeTestForced != null) { | ||||
| 			modeTest = modeTestForced; | ||||
| 		} | ||||
| 		// for local test: | ||||
| 		ConfigBaseVariable.apiAdress = "http://127.0.0.1:12342/test/api/"; | ||||
| 		// Enable the test mode permit to access to the test token (never use it in production). | ||||
| 		ConfigBaseVariable.testMode = "true"; | ||||
| 		if ("SQLITE-MEMORY".equalsIgnoreCase(modeTest)) { | ||||
| 			ConfigBaseVariable.dbType = "sqlite"; | ||||
| 			ConfigBaseVariable.bdDatabase = null; | ||||
| 			ConfigBaseVariable.dbHost = "memory"; | ||||
| 			// for test we need to connect all time the DB | ||||
| 			ConfigBaseVariable.dbKeepConnected = "true"; | ||||
| 		} else if ("SQLITE".equalsIgnoreCase(modeTest)) { | ||||
| 			ConfigBaseVariable.dbType = "sqlite"; | ||||
| 			ConfigBaseVariable.bdDatabase = null; | ||||
| 			ConfigBaseVariable.dbKeepConnected = "true"; | ||||
| 		} else if ("MY-SQL".equalsIgnoreCase(modeTest)) { | ||||
| 			ConfigBaseVariable.dbType = "mysql"; | ||||
| 			ConfigBaseVariable.bdDatabase = "test_Karideo_db"; | ||||
| 			ConfigBaseVariable.dbPort = "3906"; | ||||
| 			ConfigBaseVariable.dbUser = "root"; | ||||
| 		} else if ("MONGO".equalsIgnoreCase(modeTest)) { | ||||
| 			ConfigBaseVariable.dbType = "mongo"; | ||||
| 			ConfigBaseVariable.bdDatabase = "test_Karideo_db"; | ||||
| 		} else { | ||||
| 			// User local modification ... | ||||
| 			ConfigBaseVariable.bdDatabase = "test_Karideo_db"; | ||||
| 			ConfigBaseVariable.dbPort = "3906"; | ||||
| 			ConfigBaseVariable.dbUser = "root"; | ||||
| 		} | ||||
| 		removeDB(); | ||||
| 		// Connect the dataBase... | ||||
| 		da = DBAccess.createInterface(); | ||||
| 	} | ||||
|  | ||||
| 	public static void removeDB() { | ||||
| 		String modeTest = System.getenv("TEST_E2E_MODE"); | ||||
| 		if (modeTest == null || modeTest.isEmpty() || "false".equalsIgnoreCase(modeTest)) { | ||||
| 			modeTest = "SQLITE-MEMORY"; | ||||
| 		} else if ("true".equalsIgnoreCase(modeTest)) { | ||||
| 			modeTest = "MY-SQL"; | ||||
| 		} | ||||
| 		// override the local test: | ||||
| 		if (modeTestForced != null) { | ||||
| 			modeTest = modeTestForced; | ||||
| 		} | ||||
| 		DbConfig config = null; | ||||
| 		try { | ||||
| 			config = new DbConfig(); | ||||
| 		} catch (final DataAccessException e) { | ||||
| 			e.printStackTrace(); | ||||
| 			LOGGER.error("Fail to clean the DB"); | ||||
| 			return; | ||||
| 		} | ||||
| 		config.setDbName(null); | ||||
| 		LOGGER.info("Remove the DB and create a new one '{}'", config.getDbName()); | ||||
| 		try (final DBAccess daRoot = DBAccess.createInterface(config)) { | ||||
| 			if ("SQLITE-MEMORY".equalsIgnoreCase(modeTest)) { | ||||
| 				// nothing to do ... | ||||
| 			} else if ("SQLITE".equalsIgnoreCase(modeTest)) { | ||||
| 				daRoot.deleteDB(ConfigBaseVariable.bdDatabase); | ||||
| 			} else if ("MY-SQL".equalsIgnoreCase(modeTest)) { | ||||
| 				daRoot.deleteDB(ConfigBaseVariable.bdDatabase); | ||||
| 			} else if ("MONGO".equalsIgnoreCase(modeTest)) { | ||||
| 				daRoot.deleteDB(ConfigBaseVariable.bdDatabase); | ||||
| 			} | ||||
| 			daRoot.createDB(ConfigBaseVariable.bdDatabase); | ||||
| 		} catch (final InternalServerErrorException e) { | ||||
| 			e.printStackTrace(); | ||||
| 			LOGGER.error("Fail to clean the DB"); | ||||
| 			return; | ||||
| 		} catch (final IOException e) { | ||||
| 			e.printStackTrace(); | ||||
| 			LOGGER.error("Fail to clean the DB"); | ||||
| 			return; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public static void clear() throws IOException { | ||||
| 		LOGGER.info("Remove the test db"); | ||||
| 		removeDB(); | ||||
| 		// The connection is by default open ==> close it at the end of test: | ||||
| 		da.close(); | ||||
| 		DbIoFactory.closeAllForceMode(); | ||||
| 		ConfigBaseVariable.clearAllValue(); | ||||
| 	} | ||||
| } | ||||
| @@ -1,4 +1,4 @@ | ||||
| package test.kar.karideo; | ||||
| package test.atriasoft.karideo; | ||||
| 
 | ||||
| import org.junit.jupiter.api.extension.ConditionEvaluationResult; | ||||
| import org.junit.jupiter.api.extension.ExecutionCondition; | ||||
							
								
								
									
										48
									
								
								back/test/src/test/atriasoft/karideo/TestBase.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								back/test/src/test/atriasoft/karideo/TestBase.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| package test.atriasoft.karideo; | ||||
|  | ||||
| import org.junit.jupiter.api.AfterAll; | ||||
| import org.junit.jupiter.api.BeforeAll; | ||||
| import org.junit.jupiter.api.MethodOrderer; | ||||
| import org.junit.jupiter.api.Test; | ||||
| import org.junit.jupiter.api.TestMethodOrder; | ||||
| import org.junit.jupiter.api.extension.ExtendWith; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.archidata.tools.RESTApi; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| @ExtendWith(StepwiseExtension.class) | ||||
| @TestMethodOrder(MethodOrderer.OrderAnnotation.class) | ||||
| public class TestBase { | ||||
| 	private final static Logger LOGGER = LoggerFactory.getLogger(TestBase.class); | ||||
| 	public final static String ENDPOINT_NAME = "species/"; | ||||
|  | ||||
| 	static WebLauncherTest webInterface = null; | ||||
| 	static RESTApi api = null; | ||||
|  | ||||
| 	@BeforeAll | ||||
| 	public static void configureWebServer() throws Exception { | ||||
| 		ConfigureDb.configure(); | ||||
| 		LOGGER.info("configure server ..."); | ||||
| 		webInterface = new WebLauncherTest(); | ||||
| 		LOGGER.info("Start REST (BEGIN)"); | ||||
| 		webInterface.process(); | ||||
| 		LOGGER.info("Start REST (DONE)"); | ||||
| 		api = new RESTApi(ConfigBaseVariable.apiAdress); | ||||
| 		api.setToken(Common.ADMIN_TOKEN); | ||||
| 	} | ||||
|  | ||||
| 	@AfterAll | ||||
| 	public static void stopWebServer() throws Exception { | ||||
| 		LOGGER.info("Kill the web server"); | ||||
| 		webInterface.stop(); | ||||
| 		webInterface = null; | ||||
| 		ConfigureDb.clear(); | ||||
| 	} | ||||
|  | ||||
| 	@Test | ||||
| 	public static void TestEmpty() throws Exception { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @@ -1,7 +1,9 @@ | ||||
| package test.kar.karideo; | ||||
| 
 | ||||
| import java.io.IOException; | ||||
| package test.atriasoft.karideo; | ||||
| 
 | ||||
| import org.atriasoft.archidata.exception.RESTErrorResponseException; | ||||
| import org.atriasoft.archidata.tools.ConfigBaseVariable; | ||||
| import org.atriasoft.archidata.tools.RESTApi; | ||||
| import org.atriasoft.karideo.api.HealthCheck.HealthResult; | ||||
| import org.junit.jupiter.api.AfterAll; | ||||
| import org.junit.jupiter.api.Assertions; | ||||
| import org.junit.jupiter.api.BeforeAll; | ||||
| @@ -10,11 +12,6 @@ import org.junit.jupiter.api.Order; | ||||
| import org.junit.jupiter.api.Test; | ||||
| import org.junit.jupiter.api.TestMethodOrder; | ||||
| import org.junit.jupiter.api.extension.ExtendWith; | ||||
| import org.kar.archidata.db.DBEntry; | ||||
| import org.kar.archidata.exception.RESTErrorResponseExeption; | ||||
| import org.kar.archidata.tools.ConfigBaseVariable; | ||||
| import org.kar.archidata.tools.RESTApi; | ||||
| import org.kar.karideo.api.HealthCheck.HealthResult; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| 
 | ||||
| @@ -22,49 +19,42 @@ import org.slf4j.LoggerFactory; | ||||
| @TestMethodOrder(MethodOrderer.OrderAnnotation.class) | ||||
| public class TestHealthCheck { | ||||
| 	private final static Logger LOGGER = LoggerFactory.getLogger(TestHealthCheck.class); | ||||
| 	 | ||||
| 
 | ||||
| 	static WebLauncherTest webInterface = null; | ||||
| 	static RESTApi api = null; | ||||
| 
 | ||||
| 	@BeforeAll | ||||
| 	public static void configureWebServer() throws Exception { | ||||
| 		ConfigureDb.configure(); | ||||
| 		LOGGER.info("configure server ..."); | ||||
| 		webInterface = new WebLauncherTest(); | ||||
| 		LOGGER.info("Create DB"); | ||||
| 		try { | ||||
| 			webInterface.migrateDB(); | ||||
| 		} catch (final Exception ex) { | ||||
| 			ex.printStackTrace(); | ||||
| 			LOGGER.error("Detect an error: {}", ex.getMessage()); | ||||
| 		} | ||||
| 		LOGGER.info("Start REST (BEGIN)"); | ||||
| 		webInterface.process(); | ||||
| 		LOGGER.info("Start REST (DONE)"); | ||||
| 		api = new RESTApi(ConfigBaseVariable.apiAdress); | ||||
| 		api.setToken(Common.ADMIN_TOKEN); | ||||
| 	} | ||||
| 	 | ||||
| 
 | ||||
| 	@AfterAll | ||||
| 	public static void stopWebServer() throws InterruptedException, IOException { | ||||
| 	public static void stopWebServer() throws Exception { | ||||
| 		LOGGER.info("Kill the web server"); | ||||
| 		webInterface.stop(); | ||||
| 		webInterface = null; | ||||
| 		LOGGER.info("Remove the test db"); | ||||
| 		DBEntry.closeAllForceMode(); | ||||
| 		ConfigBaseVariable.clearAllValue(); | ||||
| 		ConfigureDb.clear(); | ||||
| 	} | ||||
| 	 | ||||
| 
 | ||||
| 	@Order(1) | ||||
| 	@Test | ||||
| 	//@RepeatedTest(10) | ||||
| 	// @RepeatedTest(10) | ||||
| 	public void checkHealthCheck() throws Exception { | ||||
| 		final HealthResult result = api.get(HealthResult.class, "health_check"); | ||||
| 		final HealthResult result = api.request("health_check").get().fetch(HealthResult.class); | ||||
| 		Assertions.assertEquals(result.value(), "alive and kicking"); | ||||
| 	} | ||||
| 	 | ||||
| 
 | ||||
| 	@Order(2) | ||||
| 	@Test | ||||
| 	public void checkHealthCheckWrongAPI() throws Exception { | ||||
| 		Assertions.assertThrows(RESTErrorResponseExeption.class, () -> api.get(HealthResult.class, "health_checks")); | ||||
| 		Assertions.assertThrows(RESTErrorResponseException.class, () -> api.request("health_check_kaboom").get().fetch()); | ||||
| 	} | ||||
| 	 | ||||
| 
 | ||||
| } | ||||
							
								
								
									
										12
									
								
								back/test/src/test/atriasoft/karideo/WebLauncherTest.java
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										12
									
								
								back/test/src/test/atriasoft/karideo/WebLauncherTest.java
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,12 @@ | ||||
|  | ||||
| package test.atriasoft.karideo; | ||||
|  | ||||
| import org.atriasoft.karideo.WebLauncher; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class WebLauncherTest extends WebLauncher { | ||||
| 	final private static Logger LOGGER = LoggerFactory.getLogger(WebLauncherTest.class); | ||||
|  | ||||
| 	public WebLauncherTest() {} | ||||
| } | ||||
| @@ -1,28 +0,0 @@ | ||||
|  | ||||
| package test.kar.karideo; | ||||
|  | ||||
| import org.kar.archidata.tools.ConfigBaseVariable; | ||||
| import org.kar.karideo.WebLauncher; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| public class WebLauncherTest extends WebLauncher { | ||||
| 	final private static Logger LOGGER = LoggerFactory.getLogger(WebLauncherTest.class); | ||||
| 	 | ||||
| 	public WebLauncherTest() { | ||||
| 		LOGGER.debug("Configure REST system"); | ||||
| 		// for local test: | ||||
| 		ConfigBaseVariable.apiAdress = "http://127.0.0.1:12345/test/api/"; | ||||
| 		ConfigBaseVariable.testMode = "true"; | ||||
| 		//ConfigBaseVariable.dbPort = "3306"; | ||||
| 		// for the test we a in memory sqlite.. | ||||
| 		ConfigBaseVariable.dbType = "sqlite"; | ||||
| 		ConfigBaseVariable.dbHost = "memory"; | ||||
| 		// for test we need to connect all time the DB | ||||
| 		ConfigBaseVariable.dbKeepConnected = "true"; | ||||
| 		 | ||||
| 		//ConfigBaseVariable.dbHost = "localhost"; | ||||
| 		//ConfigBaseVariable.dbUser = "root"; | ||||
| 		//ConfigBaseVariable.dbPassword = "ZERTYSDGFVHSDFGHJYZSDFGSQxfgsqdfgsqdrf4564654"; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										1
									
								
								front/.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								front/.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| NODE_ENV=development | ||||
							
								
								
									
										2
									
								
								front/.env.production
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								front/.env.production
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # URL for database connection | ||||
| VITE_API_BASE_URL=karideo/api/ | ||||
							
								
								
									
										27
									
								
								front/.storybook/main.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								front/.storybook/main.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| import type { StorybookConfig } from '@storybook/react-vite'; | ||||
|  | ||||
| const config: StorybookConfig = { | ||||
|   framework: { | ||||
|     name: '@storybook/react-vite', | ||||
|     options: {}, | ||||
|   }, | ||||
|  | ||||
|   core: { | ||||
|     disableTelemetry: true, | ||||
|     builder: '@storybook/builder-vite', | ||||
|   }, | ||||
|  | ||||
|   stories: ['../src/**/*.@(mdx|stories.@(js|jsx|ts|tsx))'], | ||||
|  | ||||
|   addons: ['@storybook/addon-links', '@storybook/addon-essentials'], | ||||
|  | ||||
|   staticDirs: ['../public'], | ||||
|  | ||||
|   typescript: { | ||||
|     reactDocgen: false, | ||||
|   }, | ||||
|  | ||||
|   docs: {}, | ||||
| }; | ||||
|  | ||||
| export default config; | ||||
							
								
								
									
										16
									
								
								front/.storybook/preview-head.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								front/.storybook/preview-head.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| <style> | ||||
|   html { | ||||
|     background: transparent !important; | ||||
|   } | ||||
|  | ||||
|   .docs-story > :first-child { | ||||
|     padding: 0; | ||||
|   } | ||||
|   .docs-story > * { | ||||
|     background: transparent !important; | ||||
|   } | ||||
|  | ||||
|   #root #start-ui-storybook-wrapper { | ||||
|     min-height: 100vh; | ||||
|   } | ||||
| </style> | ||||
							
								
								
									
										34
									
								
								front/.storybook/preview.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								front/.storybook/preview.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| import React from 'react'; | ||||
|  | ||||
| import { Box } from '@chakra-ui/react'; | ||||
| import { ChakraProvider } from '@chakra-ui/react'; | ||||
| import { MemoryRouter } from 'react-router-dom'; | ||||
|  | ||||
| import { ColorModeProvider } from '../src/components/ui/color-mode'; | ||||
| import { Toaster } from '../src/components/ui/toaster'; | ||||
| import { systemTheme } from '../src/theme/theme'; | ||||
|  | ||||
| // . | ||||
| const DocumentationWrapper = ({ children }) => { | ||||
|   return ( | ||||
|     <Box id="start-ui-storybook-wrapper" p="4" pb="8" flex="1"> | ||||
|       {children} | ||||
|     </Box> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export const decorators = [ | ||||
|   (Story, context) => ( | ||||
|     <ColorModeProvider> | ||||
|       <ChakraProvider value={systemTheme}> | ||||
|         {/* Using MemoryRouter to avoid route clashing with Storybook */} | ||||
|         <MemoryRouter> | ||||
|           <DocumentationWrapper> | ||||
|             <Story {...context} /> | ||||
|           </DocumentationWrapper> | ||||
|         </MemoryRouter> | ||||
|         <Toaster /> | ||||
|       </ChakraProvider> | ||||
|     </ColorModeProvider> | ||||
|   ), | ||||
| ]; | ||||
							
								
								
									
										2
									
								
								front/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								front/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| Proprietary | ||||
| @copyright Edouard Dupin 2024 | ||||
							
								
								
									
										10637
									
								
								front/config sample.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10637
									
								
								front/config sample.yaml
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										14
									
								
								front/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								front/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="UTF-8" /> | ||||
|     <link rel="manifest" href="/manifest.json" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|     <title>Karideo</title> | ||||
|     <link rel="icon" href="/favicon.ico" /> | ||||
|   </head> | ||||
|   <body style="width:100vw;height:100vh;min-width:100%;min-height:100%;"> | ||||
|     <div id="root" style="width:100%;height:100%;min-width:100%;min-height:100%;"></div> | ||||
|     <script type="module" src="/src/main.tsx"></script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										9
									
								
								front/knip.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								front/knip.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| import type { KnipConfig } from 'knip'; | ||||
|  | ||||
| const config: KnipConfig = { | ||||
|   // Ignoring mostly shell binaries | ||||
|   ignoreBinaries: ['export', 'sleep'], | ||||
|   ignore: [], | ||||
| }; | ||||
|  | ||||
| export default config; | ||||
| @@ -1,50 +1,89 @@ | ||||
| { | ||||
| 	"name": "karideo", | ||||
| 	"version": "0.0.0", | ||||
| 	"license": "MPL-2", | ||||
| 	"scripts": { | ||||
| 		"all": "npm run build && npm run test", | ||||
| 		"ng": "ng", | ||||
| 		"dev": "ng serve --configuration=develop --watch --port 4202", | ||||
| 		"dev-hot-update": "ng serve --configuration=develop --watch --hmr --port 4202", | ||||
| 		"build": "ng build --prod", | ||||
| 		"test": "ng test", | ||||
| 		"lint": "ng lint", | ||||
| 		"style": "prettier --write .", | ||||
| 		"e2e": "ng e2e", | ||||
| 		"update_packages": "ncu --upgrade", | ||||
| 		"install_dependency": "pnpm install --force", | ||||
| 		"link_kar_cw": "pnpm link ../../kar-cw/dist/kar-cw/", | ||||
| 		"unlink_kar_cw": "pnpm unlink ../../kar-cw/dist/kar-cw/" | ||||
| 	}, | ||||
| 	"private": true, | ||||
| 	"dependencies": { | ||||
| 		"@angular/animations": "^17.3.5", | ||||
| 		"@angular/cdk": "^17.3.5", | ||||
| 		"@angular/common": "^17.3.5", | ||||
| 		"@angular/compiler": "^17.3.5", | ||||
| 		"@angular/core": "^17.3.5", | ||||
| 		"@angular/forms": "^17.3.5", | ||||
| 		"@angular/material": "^17.3.5", | ||||
| 		"@angular/platform-browser": "^17.3.5", | ||||
| 		"@angular/platform-browser-dynamic": "^17.3.5", | ||||
| 		"@angular/router": "^17.3.5", | ||||
| 		"rxjs": "^7.8.1", | ||||
| 		"zone.js": "^0.14.4", | ||||
| 		"zod": "3.22.4", | ||||
| 		"@kangaroo-and-rabbit/kar-cw": "^0.2.1" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@angular-devkit/build-angular": "^17.3.5", | ||||
| 		"@angular-eslint/builder": "17.3.0", | ||||
| 		"@angular-eslint/eslint-plugin": "17.3.0", | ||||
| 		"@angular-eslint/eslint-plugin-template": "17.3.0", | ||||
| 		"@angular-eslint/schematics": "17.3.0", | ||||
| 		"@angular-eslint/template-parser": "17.3.0", | ||||
| 		"@angular/cli": "^17.3.5", | ||||
| 		"@angular/compiler-cli": "^17.3.5", | ||||
| 		"@angular/language-service": "^17.3.5", | ||||
| 		"npm-check-updates": "^16.14.18", | ||||
| 		"tslib": "^2.6.2" | ||||
| 	} | ||||
| } | ||||
|   "name": "karideo", | ||||
|   "private": true, | ||||
|   "version": "0.0.1", | ||||
|   "description": "KAR web music application", | ||||
|   "author": { | ||||
|     "name": "Edouard DUPIN", | ||||
|     "email": "yui.heero@gmail.farm" | ||||
|   }, | ||||
|   "license": "PROPRIETARY", | ||||
|   "engines": { | ||||
|     "node": ">=20" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "update_packages": "ncu --target minor", | ||||
|     "upgrade_packages": "ncu --upgrade ", | ||||
|     "install_dependency": "pnpm install", | ||||
|     "test": "vitest run", | ||||
|     "test:watch": "vitest watch", | ||||
|     "build": "tsc && vite build", | ||||
|     "static:build": "pnpm build", | ||||
|     "dev": "vite", | ||||
|     "pretty": "prettier -w .", | ||||
|     "lint": "pnpm tsc --noEmit", | ||||
|     "storybook": "storybook dev -p 3001", | ||||
|     "storybook:build": "storybook build && mv ./storybook-static ./public/storybook" | ||||
|   }, | ||||
|   "lint-staged": { | ||||
|     "*.{ts,tsx,js,jsx,json}": "prettier --write" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "react-speech-recognition": "4.0.1", | ||||
|     "regenerator-runtime": "0.14.1", | ||||
|     "@locator/babel-jsx": "0.4.4", | ||||
|     "@trivago/prettier-plugin-sort-imports": "5.2.2", | ||||
|     "@chakra-ui/cli": "3.27.1", | ||||
|     "@chakra-ui/react": "3.27.1", | ||||
|     "@emotion/react": "11.14.0", | ||||
|     "allotment": "1.20.4", | ||||
|     "css-mediaquery": "0.1.2", | ||||
|     "dayjs": "1.11.18", | ||||
|     "history": "5.3.0", | ||||
|     "next-themes": "^0.4.6", | ||||
|     "react": "19.2.0", | ||||
|     "react-dom": "19.2.0", | ||||
|     "react-error-boundary": "6.0.0", | ||||
|     "react-icons": "5.5.0", | ||||
|     "react-router-dom": "7.9.4", | ||||
|     "react-select": "5.10.2", | ||||
|     "react-use": "17.6.0", | ||||
|     "zod": "4.1.12", | ||||
|     "zustand": "5.0.8" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@chakra-ui/styled-system": "^2.12.0", | ||||
|     "@playwright/test": "1.56.1", | ||||
|     "@storybook/addon-links": "9.1.13", | ||||
|     "@storybook/react-vite": "9.1.13", | ||||
|     "@testing-library/jest-dom": "6.9.1", | ||||
|     "@testing-library/react": "16.3.0", | ||||
|     "@testing-library/user-event": "14.6.1", | ||||
|     "@trivago/prettier-plugin-sort-imports": "5.2.2", | ||||
|     "@types/jest": "30.0.0", | ||||
|     "@types/node": "24.8.1", | ||||
|     "@types/react": "19.2.2", | ||||
|     "@types/react-dom": "19.2.2", | ||||
|     "@typescript-eslint/eslint-plugin": "8.46.1", | ||||
|     "@typescript-eslint/parser": "8.46.1", | ||||
|     "@vitejs/plugin-react": "5.0.4", | ||||
|     "eslint": "9.38.0", | ||||
|     "eslint-plugin-import": "2.32.0", | ||||
|     "eslint-plugin-react": "7.37.5", | ||||
|     "eslint-plugin-react-hooks": "7.0.0", | ||||
|     "eslint-plugin-storybook": "9.1.13", | ||||
|     "jest": "30.2.0", | ||||
|     "jest-environment-jsdom": "30.2.0", | ||||
|     "knip": "5.66.0", | ||||
|     "lint-staged": "16.2.4", | ||||
|     "npm-check-updates": "^19.1.1", | ||||
|     "prettier": "3.6.2", | ||||
|     "puppeteer": "24.25.0", | ||||
|     "react-is": "19.2.0", | ||||
|     "storybook": "9.1.13", | ||||
|     "ts-node": "10.9.2", | ||||
|     "typescript": "5.9.3", | ||||
|     "vite": "7.1.10", | ||||
|     "vitest": "3.2.4" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										18969
									
								
								front/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										18969
									
								
								front/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										16
									
								
								front/prettier.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								front/prettier.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| // Using a JS file, allowing us to add comments | ||||
| module.exports = { | ||||
|   // This plugins line is mandatory for the plugin to work with pnpm. | ||||
|   // https://github.com/trivago/prettier-plugin-sort-imports/blob/61d069711008c530f5a41ca4e254781abc5de358/README.md?plain=1#L89-L96 | ||||
|   plugins: ['@trivago/prettier-plugin-sort-imports'], | ||||
|   endOfLine: 'lf', | ||||
|   semi: true, | ||||
|   singleQuote: true, | ||||
|   tabWidth: 2, | ||||
|   trailingComma: 'es5', | ||||
|   arrowParens: 'always', | ||||
|   importOrder: ['^react$', '^(?!^react$|^@/|^[./]).*', '^@/(.*)$', '^[./]'], | ||||
|   importOrderSeparation: true, | ||||
|   importOrderSortSpecifiers: true, | ||||
|   importOrderParserPlugins: ['jsx', 'typescript'], | ||||
| }; | ||||
| Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										
											BIN
										
									
								
								front/public/icons/icon-192x192.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								front/public/icons/icon-192x192.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										
											BIN
										
									
								
								front/public/icons/icon-512x512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								front/public/icons/icon-512x512.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 41 KiB | 
							
								
								
									
										21
									
								
								front/public/manifest.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								front/public/manifest.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| { | ||||
|   "name": "Karusic", | ||||
|   "short_name": "Karusic", | ||||
|   "description": "(K)angaroo (A)nd (R)abbit m(usic) is a music streaming", | ||||
|   "start_url": "/karusic/", | ||||
|   "display": "standalone", | ||||
|   "background_color": "#000000", | ||||
|   "theme_color": "#FFFFFF", | ||||
|   "icons": [ | ||||
|     { | ||||
|       "src": "/karusic/icons/icon-192x192.png", | ||||
|       "sizes": "192x192", | ||||
|       "type": "image/png" | ||||
|     }, | ||||
|     { | ||||
|       "src": "/karusic/icons/icon-512x512.png", | ||||
|       "sizes": "512x512", | ||||
|       "type": "image/png" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										20
									
								
								front/src/App.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								front/src/App.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| import { ErrorBoundary } from '@/errors/ErrorBoundary'; | ||||
|  | ||||
| import { VideoPlayer } from './components'; | ||||
| import { EnvDevelopment } from './components/EnvDevelopment/EnvDevelopment'; | ||||
| import { AppRoutes } from './scene/AppRoutes'; | ||||
| import { ServiceContextProvider } from './service/ServiceContext'; | ||||
|  | ||||
| export const App = () => { | ||||
|   return ( | ||||
|     <ServiceContextProvider> | ||||
|       <EnvDevelopment /> | ||||
|       <ErrorBoundary> | ||||
|         <AppRoutes /> | ||||
|       </ErrorBoundary> | ||||
|       <VideoPlayer /> | ||||
|     </ServiceContextProvider> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default App; | ||||
| @@ -1,8 +0,0 @@ | ||||
| /** | ||||
|  * API of the server (auto-generated code) | ||||
|  */ | ||||
| import { HTTPMimeType, HTTPRequestModel, ModelResponseHttp, RESTConfig, RESTCallbacks, RESTRequestJson, RESTRequestJsonArray, RESTRequestVoid } from "./rest-tools" | ||||
| import {} from "./model" | ||||
| export namespace Front { | ||||
|  | ||||
| } | ||||
| @@ -1,23 +0,0 @@ | ||||
| /** | ||||
|  * API of the server (auto-generated code) | ||||
|  */ | ||||
| import { HTTPMimeType, HTTPRequestModel, ModelResponseHttp, RESTConfig, RESTCallbacks, RESTRequestJson, RESTRequestJsonArray, RESTRequestVoid } from "./rest-tools" | ||||
| import {HealthResult, isHealthResult, } from "./model" | ||||
| export namespace HealthCheck { | ||||
|  | ||||
| 	/** | ||||
| 	 * Get the server state (health) | ||||
| 	 */ | ||||
| 	export function getHealth({ restConfig, }: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<HealthResult> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/health_check", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isHealthResult); | ||||
| 	}; | ||||
| } | ||||
| @@ -1,13 +0,0 @@ | ||||
| /** | ||||
|  * Global import of the package | ||||
|  */ | ||||
| export * from "./model"; | ||||
| export * from "./front"; | ||||
| export * from "./health-check"; | ||||
| export * from "./season-resource"; | ||||
| export * from "./series-resource"; | ||||
| export * from "./type-resource"; | ||||
| export * from "./user-media-advancement-resource"; | ||||
| export * from "./user-resource"; | ||||
| export * from "./media-resource"; | ||||
| export * from "./data-resource"; | ||||
| @@ -1,435 +0,0 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { z as zod } from "zod"; | ||||
|  | ||||
| export const ZodUUID = zod.string().uuid(); | ||||
| export type UUID = zod.infer<typeof ZodUUID>; | ||||
| export function isUUID(data: any): data is UUID { | ||||
| 	try { | ||||
| 		ZodUUID.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodLong = zod.number(); | ||||
| export type Long = zod.infer<typeof ZodLong>; | ||||
| export function isLong(data: any): data is Long { | ||||
| 	try { | ||||
| 		ZodLong.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodInteger = zod.number().safe(); | ||||
| export type Integer = zod.infer<typeof ZodInteger>; | ||||
| export function isInteger(data: any): data is Integer { | ||||
| 	try { | ||||
| 		ZodInteger.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodFloat = zod.number(); | ||||
| export type Float = zod.infer<typeof ZodFloat>; | ||||
| export function isFloat(data: any): data is Float { | ||||
| 	try { | ||||
| 		ZodFloat.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodInstant = zod.string(); | ||||
| export type Instant = zod.infer<typeof ZodInstant>; | ||||
| export function isInstant(data: any): data is Instant { | ||||
| 	try { | ||||
| 		ZodInstant.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodDate = zod.date(); | ||||
| export type Date = zod.infer<typeof ZodDate>; | ||||
| export function isDate(data: any): data is Date { | ||||
| 	try { | ||||
| 		ZodDate.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodTimestamp = zod.date(); | ||||
| export type Timestamp = zod.infer<typeof ZodTimestamp>; | ||||
| export function isTimestamp(data: any): data is Timestamp { | ||||
| 	try { | ||||
| 		ZodTimestamp.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodLocalDate = zod.date(); | ||||
| export type LocalDate = zod.infer<typeof ZodLocalDate>; | ||||
| export function isLocalDate(data: any): data is LocalDate { | ||||
| 	try { | ||||
| 		ZodLocalDate.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodLocalTime = zod.date(); | ||||
| export type LocalTime = zod.infer<typeof ZodLocalTime>; | ||||
| export function isLocalTime(data: any): data is LocalTime { | ||||
| 	try { | ||||
| 		ZodLocalTime.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodRestErrorResponse = zod.object({ | ||||
| 	uuid: ZodUUID.optional(), | ||||
| 	name: zod.string().max(255).optional(), | ||||
| 	message: zod.string().max(255).optional(), | ||||
| 	time: zod.string().max(255).optional(), | ||||
| 	status: ZodInteger, | ||||
| 	statusMessage: zod.string().max(255).optional() | ||||
| }); | ||||
| export type RestErrorResponse = zod.infer<typeof ZodRestErrorResponse>; | ||||
| export function isRestErrorResponse(data: any): data is RestErrorResponse { | ||||
| 	try { | ||||
| 		ZodRestErrorResponse.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodHealthResult = zod.object({ | ||||
| }); | ||||
| export type HealthResult = zod.infer<typeof ZodHealthResult>; | ||||
| export function isHealthResult(data: any): data is HealthResult { | ||||
| 	try { | ||||
| 		ZodHealthResult.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodGenericTiming = zod.object({ | ||||
| 	// Create time of the object | ||||
| 	createdAt: ZodDate.readonly().optional(), | ||||
| 	// When update the object | ||||
| 	updatedAt: ZodDate.readonly().optional() | ||||
| }); | ||||
| export type GenericTiming = zod.infer<typeof ZodGenericTiming>; | ||||
| export function isGenericTiming(data: any): data is GenericTiming { | ||||
| 	try { | ||||
| 		ZodGenericTiming.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodGenericData = ZodGenericTiming.extend({ | ||||
| 	// Unique Id of the object | ||||
| 	id: ZodLong.readonly().optional() | ||||
| }); | ||||
| export type GenericData = zod.infer<typeof ZodGenericData>; | ||||
| export function isGenericData(data: any): data is GenericData { | ||||
| 	try { | ||||
| 		ZodGenericData.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodGenericDataSoftDelete = ZodGenericData.extend({ | ||||
| 	// Deleted state | ||||
| 	deleted: zod.boolean().readonly().optional() | ||||
| }); | ||||
| export type GenericDataSoftDelete = zod.infer<typeof ZodGenericDataSoftDelete>; | ||||
| export function isGenericDataSoftDelete(data: any): data is GenericDataSoftDelete { | ||||
| 	try { | ||||
| 		ZodGenericDataSoftDelete.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodSeason = ZodGenericDataSoftDelete.extend({ | ||||
| 	// Name of the media (this represent the title) | ||||
| 	name: zod.string().optional(), | ||||
| 	// Description of the media | ||||
| 	description: zod.string().optional(), | ||||
| 	// series parent ID | ||||
| 	parentId: ZodLong.optional(), | ||||
| 	// List of Id of the specific covers | ||||
| 	covers: zod.array(ZodUUID).optional() | ||||
| }); | ||||
| export type Season = zod.infer<typeof ZodSeason>; | ||||
| export function isSeason(data: any): data is Season { | ||||
| 	try { | ||||
| 		ZodSeason.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodSeries = ZodGenericDataSoftDelete.extend({ | ||||
| 	// Name of the media (this represent the title) | ||||
| 	name: zod.string().optional(), | ||||
| 	// Description of the media | ||||
| 	description: zod.string().optional(), | ||||
| 	// series parent ID | ||||
| 	parentId: ZodLong.optional(), | ||||
| 	// List of Id of the specific covers | ||||
| 	covers: zod.array(ZodUUID).optional() | ||||
| }); | ||||
| export type Series = zod.infer<typeof ZodSeries>; | ||||
| export function isSeries(data: any): data is Series { | ||||
| 	try { | ||||
| 		ZodSeries.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodType = ZodGenericDataSoftDelete.extend({ | ||||
| 	// Name of the media (this represent the title) | ||||
| 	name: zod.string().optional(), | ||||
| 	// Description of the media | ||||
| 	description: zod.string().optional(), | ||||
| 	// List of Id of the specific covers | ||||
| 	covers: zod.array(ZodUUID).optional() | ||||
| }); | ||||
| export type Type = zod.infer<typeof ZodType>; | ||||
| export function isType(data: any): data is Type { | ||||
| 	try { | ||||
| 		ZodType.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodUserMediaAdvancement = ZodGenericDataSoftDelete.extend({ | ||||
| 	// Foreign Key Id of the user | ||||
| 	userId: ZodLong.optional(), | ||||
| 	// Id of the media | ||||
| 	mediaId: ZodLong.optional(), | ||||
| 	// Percent of advancement in the media | ||||
| 	percent: ZodFloat.optional(), | ||||
| 	// Number of second of advancement in the media | ||||
| 	time: ZodInteger.optional(), | ||||
| 	// Number of time this media has been read | ||||
| 	count: ZodInteger.optional() | ||||
| }); | ||||
| export type UserMediaAdvancement = zod.infer<typeof ZodUserMediaAdvancement>; | ||||
| export function isUserMediaAdvancement(data: any): data is UserMediaAdvancement { | ||||
| 	try { | ||||
| 		ZodUserMediaAdvancement.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodMediaInformationsDelta = zod.object({ | ||||
| }); | ||||
| export type MediaInformationsDelta = zod.infer<typeof ZodMediaInformationsDelta>; | ||||
| export function isMediaInformationsDelta(data: any): data is MediaInformationsDelta { | ||||
| 	try { | ||||
| 		ZodMediaInformationsDelta.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodUser = ZodGenericDataSoftDelete.extend({ | ||||
| 	login: zod.string().max(128).optional(), | ||||
| 	lastConnection: ZodTimestamp.optional(), | ||||
| 	admin: zod.boolean(), | ||||
| 	blocked: zod.boolean(), | ||||
| 	removed: zod.boolean(), | ||||
| 	covers: zod.array(ZodLong).optional() | ||||
| }); | ||||
| export type User = zod.infer<typeof ZodUser>; | ||||
| export function isUser(data: any): data is User { | ||||
| 	try { | ||||
| 		ZodUser.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodUserKarideo = ZodUser.extend({ | ||||
| }); | ||||
| export type UserKarideo = zod.infer<typeof ZodUserKarideo>; | ||||
| export function isUserKarideo(data: any): data is UserKarideo { | ||||
| 	try { | ||||
| 		ZodUserKarideo.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodUserOut = zod.object({ | ||||
| 	id: ZodLong, | ||||
| 	login: zod.string().max(255).optional() | ||||
| }); | ||||
| export type UserOut = zod.infer<typeof ZodUserOut>; | ||||
| export function isUserOut(data: any): data is UserOut { | ||||
| 	try { | ||||
| 		ZodUserOut.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodMedia = ZodGenericDataSoftDelete.extend({ | ||||
| 	name: zod.string().optional(), | ||||
| 	description: zod.string().optional(), | ||||
| 	dataId: ZodUUID.optional(), | ||||
| 	typeId: ZodLong.optional(), | ||||
| 	seriesId: ZodLong.optional(), | ||||
| 	seasonId: ZodLong.optional(), | ||||
| 	episode: ZodInteger.optional(), | ||||
| 	date: ZodInteger.optional(), | ||||
| 	time: ZodInteger.optional(), | ||||
| 	ageLimit: ZodInteger.optional(), | ||||
| 	covers: zod.array(ZodUUID).optional() | ||||
| }); | ||||
| export type Media = zod.infer<typeof ZodMedia>; | ||||
| export function isMedia(data: any): data is Media { | ||||
| 	try { | ||||
| 		ZodMedia.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodUUIDGenericData = ZodGenericTiming.extend({ | ||||
| 	// Unique UUID of the object | ||||
| 	id: ZodUUID.readonly().optional() | ||||
| }); | ||||
| export type UUIDGenericData = zod.infer<typeof ZodUUIDGenericData>; | ||||
| export function isUUIDGenericData(data: any): data is UUIDGenericData { | ||||
| 	try { | ||||
| 		ZodUUIDGenericData.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodUUIDGenericDataSoftDelete = ZodUUIDGenericData.extend({ | ||||
| 	// Deleted state | ||||
| 	deleted: zod.boolean().readonly().optional() | ||||
| }); | ||||
| export type UUIDGenericDataSoftDelete = zod.infer<typeof ZodUUIDGenericDataSoftDelete>; | ||||
| export function isUUIDGenericDataSoftDelete(data: any): data is UUIDGenericDataSoftDelete { | ||||
| 	try { | ||||
| 		ZodUUIDGenericDataSoftDelete.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| export const ZodData = ZodUUIDGenericDataSoftDelete.extend({ | ||||
| 	// Sha512 of the data | ||||
| 	sha512: zod.string().max(128).optional(), | ||||
| 	// Mime -type of the media | ||||
| 	mimeType: zod.string().max(128).optional(), | ||||
| 	// Size in Byte of the data | ||||
| 	size: ZodLong.optional() | ||||
| }); | ||||
| export type Data = zod.infer<typeof ZodData>; | ||||
| export function isData(data: any): data is Data { | ||||
| 	try { | ||||
| 		ZodData.parse(data); | ||||
| 		return true; | ||||
| 	} catch (e: any) { | ||||
| 		console.log(`Fail to parse data ${e}`); | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -1,373 +0,0 @@ | ||||
| /** @file | ||||
|  * @author Edouard DUPIN | ||||
|  * @copyright 2024, Edouard DUPIN, all right reserved | ||||
|  * @license MPL-2 | ||||
|  */ | ||||
|  | ||||
| import { RestErrorResponse } from "./model" | ||||
|  | ||||
| export enum HTTPRequestModel { | ||||
|     DELETE = 'DELETE', | ||||
|     GET = 'GET', | ||||
|     PATCH = 'PATCH', | ||||
|     POST = 'POST', | ||||
|     PUT = 'PUT', | ||||
| } | ||||
| export enum HTTPMimeType { | ||||
|     ALL = '*/*', | ||||
|     CSV = 'text/csv', | ||||
|     IMAGE = 'image/*', | ||||
|     IMAGE_JPEG = 'image/jpeg', | ||||
|     IMAGE_PNG = 'image/png', | ||||
|     JSON = 'application/json', | ||||
|     MULTIPART = 'multipart/form-data', | ||||
|     OCTET_STREAM = 'application/octet-stream', | ||||
|     TEXT_PLAIN = 'text/plain', | ||||
| } | ||||
|  | ||||
| export interface RESTConfig { | ||||
|     // base of the server: http(s)://my.server.org/plop/api/ | ||||
|     server: string; | ||||
|     // Token to access of the data. | ||||
|     token?: string; | ||||
| } | ||||
|  | ||||
| export interface RESTModel { | ||||
|     // base of the local API request: "sheep/{id}". | ||||
|     endPoint: string; | ||||
|     // Type of the request. | ||||
|     requestType?: HTTPRequestModel; | ||||
|     // Input type requested. | ||||
|     accept?: HTTPMimeType; | ||||
|     // Content of the local data. | ||||
|     contentType?: HTTPMimeType; | ||||
|     // Mode of the TOKEN in URL or Header (?token:${tokenInUrl}) | ||||
|     tokenInUrl?: boolean; | ||||
| } | ||||
|  | ||||
| export interface ModelResponseHttp { | ||||
|     status: number; | ||||
|     data: any; | ||||
| } | ||||
|  | ||||
| export function isArrayOf<TYPE>( | ||||
|     data: any, | ||||
|     typeChecker: (subData: any) => subData is TYPE, | ||||
|     length?: number | ||||
| ): data is TYPE[] { | ||||
|     if (!Array.isArray(data)) { | ||||
|         return false; | ||||
|     } | ||||
|     if (!data.every(typeChecker)) { | ||||
|         return false; | ||||
|     } | ||||
|     if (length !== undefined && data.length != length) { | ||||
|         return false; | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| function isNullOrUndefined(data: any): data is undefined | null { | ||||
|     return data === undefined || data === null; | ||||
| } | ||||
|  | ||||
| // generic progression callback | ||||
| export type ProgressCallback = (count: number, total: number) => void; | ||||
|  | ||||
| export interface RESTAbort { | ||||
|     abort?: () => boolean | ||||
| } | ||||
|  | ||||
|  | ||||
| // Rest generic callback have a basic model to upload and download advancement. | ||||
| export interface RESTCallbacks { | ||||
|     progressUpload?: ProgressCallback, | ||||
|     progressDownload?: ProgressCallback, | ||||
|     abortHandle?: RESTAbort, | ||||
| }; | ||||
|  | ||||
| export interface RESTRequestType { | ||||
|     restModel: RESTModel, | ||||
|     restConfig: RESTConfig, | ||||
|     data?: any, | ||||
|     params?: object, | ||||
|     queries?: object, | ||||
|     callback?: RESTCallbacks, | ||||
| }; | ||||
|  | ||||
| function removeTrailingSlashes(input: string): string { | ||||
|     if (isNullOrUndefined(input)) { | ||||
|         return "undefined"; | ||||
|     } | ||||
|     return input.replace(/\/+$/, ''); | ||||
| } | ||||
| function removeLeadingSlashes(input: string): string { | ||||
|     if (isNullOrUndefined(input)) { | ||||
|         return ""; | ||||
|     } | ||||
|     return input.replace(/^\/+/, ''); | ||||
| } | ||||
|  | ||||
| export function RESTUrl({ restModel, restConfig, params, queries }: RESTRequestType): string { | ||||
|     // Create the URL PATH: | ||||
|     let generateUrl = `${removeTrailingSlashes(restConfig.server)}/${removeLeadingSlashes(restModel.endPoint)}`; | ||||
|     if (params !== undefined) { | ||||
|         for (let key of Object.keys(params)) { | ||||
|             generateUrl = generateUrl.replaceAll(`{${key}}`, `${params[key]}`); | ||||
|         } | ||||
|     } | ||||
|     if (queries === undefined && (restConfig.token === undefined || restModel.tokenInUrl !== true)) { | ||||
|         return generateUrl; | ||||
|     } | ||||
|     const searchParams = new URLSearchParams(); | ||||
|     if (queries !== undefined) { | ||||
|         for (let key of Object.keys(queries)) { | ||||
|             const value = queries[key]; | ||||
|             if (Array.isArray(value)) { | ||||
|                 for (let iii = 0; iii < value.length; iii++) { | ||||
|                     searchParams.append(`${key}`, `${value[iii]}`); | ||||
|                 } | ||||
|             } else { | ||||
|                 searchParams.append(`${key}`, `${value}`); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     if (restConfig.token !== undefined && restModel.tokenInUrl === true) { | ||||
|         searchParams.append('Authorization', `Bearer ${restConfig.token}`); | ||||
|     } | ||||
|     return generateUrl + "?" + searchParams.toString(); | ||||
| } | ||||
|  | ||||
|  | ||||
| export function fetchProgress(generateUrl: string, { method, headers, body }: { | ||||
|     method: HTTPRequestModel, | ||||
|     headers: any, | ||||
|     body: any, | ||||
| }, { progressUpload, progressDownload, abortHandle }: RESTCallbacks): Promise<Response> { | ||||
|     const xhr = { | ||||
|         io: new XMLHttpRequest() | ||||
|     } | ||||
|     return new Promise((resolve, reject) => { | ||||
|         // Stream the upload progress | ||||
|         if (progressUpload) { | ||||
|             xhr.io.upload.addEventListener("progress", (dataEvent) => { | ||||
|                 if (dataEvent.lengthComputable) { | ||||
|                     //console.log(`    ==> has a progress event: ${dataEvent.loaded} / ${dataEvent.total}`); | ||||
|                     progressUpload(dataEvent.loaded, dataEvent.total); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|         // Stream the download progress | ||||
|         if (progressDownload) { | ||||
|             xhr.io.addEventListener("progress", (dataEvent) => { | ||||
|                 if (dataEvent.lengthComputable) { | ||||
|                     //console.log(`    ==> download progress:: ${dataEvent.loaded} / ${dataEvent.total}`); | ||||
|                     progressUpload(dataEvent.loaded, dataEvent.total); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|         if (abortHandle) { | ||||
|             abortHandle.abort = () => { | ||||
|                 if (xhr.io) { | ||||
|                     console.log(`Request abort on the XMLHttpRequest: ${generateUrl}`); | ||||
|                     xhr.io.abort(); | ||||
|                     return true; | ||||
|                 } | ||||
|                 console.log(`Request abort (FAIL) on the XMLHttpRequest: ${generateUrl}`); | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         // Check if we have an internal Fail: | ||||
|         xhr.io.addEventListener('error', () => { | ||||
|             xhr.io = undefined; | ||||
|             reject(new TypeError('Failed to fetch')) | ||||
|         }); | ||||
|  | ||||
|         // Capture the end of the stream | ||||
|         xhr.io.addEventListener("loadend", () => { | ||||
|             if (xhr.io.readyState !== XMLHttpRequest.DONE) { | ||||
|                 //console.log(`    ==> READY state`); | ||||
|                 return; | ||||
|             } | ||||
|             if (xhr.io.status === 0) { | ||||
|                 //the stream has been aborted | ||||
|                 reject(new TypeError('Fetch has been aborted')); | ||||
|                 return; | ||||
|             } | ||||
|             // Stream is ended, transform in a generic response: | ||||
|             const response = new Response(xhr.io.response, { | ||||
|                 status: xhr.io.status, | ||||
|                 statusText: xhr.io.statusText | ||||
|             }); | ||||
|             const headersArray = xhr.io.getAllResponseHeaders().trim().replaceAll("\r\n", "\n").split('\n'); | ||||
|             headersArray.forEach(function (header) { | ||||
|                 const firstColonIndex = header.indexOf(':'); | ||||
|                 if (firstColonIndex !== -1) { | ||||
|                     var key = header.substring(0, firstColonIndex).trim(); | ||||
|                     var value = header.substring(firstColonIndex + 1).trim(); | ||||
|                     response.headers.set(key, value); | ||||
|                 } else { | ||||
|                     response.headers.set(header, ""); | ||||
|                 } | ||||
|             }); | ||||
|             xhr.io = undefined; | ||||
|             resolve(response); | ||||
|         }); | ||||
|         xhr.io.open(method, generateUrl, true); | ||||
|         if (!isNullOrUndefined(headers)) { | ||||
|             for (const [key, value] of Object.entries(headers)) { | ||||
|                 xhr.io.setRequestHeader(key, value as string); | ||||
|             } | ||||
|         } | ||||
|         xhr.io.send(body); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| export function RESTRequest({ restModel, restConfig, data, params, queries, callback }: RESTRequestType): Promise<ModelResponseHttp> { | ||||
|     // Create the URL PATH: | ||||
|     let generateUrl = RESTUrl({ restModel, restConfig, data, params, queries }); | ||||
|     let headers: any = {}; | ||||
|     if (restConfig.token !== undefined && restModel.tokenInUrl !== true) { | ||||
|         headers['Authorization'] = `Bearer ${restConfig.token}`; | ||||
|     } | ||||
|     if (restModel.accept !== undefined) { | ||||
|         headers['Accept'] = restModel.accept; | ||||
|     } | ||||
|     if (restModel.requestType !== HTTPRequestModel.GET) { | ||||
|         // if Get we have not a content type, the body is empty | ||||
|         if (restModel.contentType !== HTTPMimeType.MULTIPART) { | ||||
|             // special case of multi-part ==> no content type otherwise the browser does not set the ";bundary=--****" | ||||
|             headers['Content-Type'] = restModel.contentType; | ||||
|         } | ||||
|     } | ||||
|     let body = data; | ||||
|     if (restModel.contentType === HTTPMimeType.JSON) { | ||||
|         body = JSON.stringify(data); | ||||
|     } else if (restModel.contentType === HTTPMimeType.MULTIPART) { | ||||
|         const formData = new FormData(); | ||||
|         for (const name in data) { | ||||
|             formData.append(name, data[name]); | ||||
|         } | ||||
|         body = formData | ||||
|     } | ||||
|     return new Promise((resolve, reject) => { | ||||
|         let action: undefined | Promise<Response> = undefined; | ||||
|         if (isNullOrUndefined(callback) | ||||
|             || (isNullOrUndefined(callback.progressDownload) | ||||
|                 && isNullOrUndefined(callback.progressUpload) | ||||
|                 && isNullOrUndefined(callback.abortHandle))) { | ||||
|             // No information needed: call the generic fetch interface | ||||
|             action = fetch(generateUrl, { | ||||
|                 method: restModel.requestType, | ||||
|                 headers, | ||||
|                 body, | ||||
|             }); | ||||
|         } else { | ||||
|             // need progression information: call old fetch model (XMLHttpRequest) that permit to keep % upload and % download for HTTP1.x | ||||
|             action = fetchProgress(generateUrl, { | ||||
|                 method: restModel.requestType ?? HTTPRequestModel.GET, | ||||
|                 headers, | ||||
|                 body, | ||||
|             }, callback); | ||||
|         } | ||||
|         action.then((response: Response) => { | ||||
|             if (response.status >= 200 && response.status <= 299) { | ||||
|                 const contentType = response.headers.get('Content-Type'); | ||||
|                 if (restModel.accept !== contentType) { | ||||
|                     reject({ | ||||
|                         time: Date().toString(), | ||||
|                         status: 901, | ||||
|                         error: `REST check wrong type: ${restModel.accept} != ${contentType}`, | ||||
|                         statusMessage: "Fetch error", | ||||
|                         message: "rest-tools.ts Wrong type in the message return type" | ||||
|                     } as RestErrorResponse); | ||||
|                 } else if (contentType === HTTPMimeType.JSON) { | ||||
|                     response | ||||
|                         .json() | ||||
|                         .then((value: any) => { | ||||
|                             //console.log(`RECEIVE ==> ${response.status}=${ JSON.stringify(value, null, 2)}`); | ||||
|                             resolve({ status: response.status, data: value }); | ||||
|                         }) | ||||
|                         .catch((reason: any) => { | ||||
|                             reject({ | ||||
|                                 time: Date().toString(), | ||||
|                                 status: 902, | ||||
|                                 error: `REST parse json fail: ${reason}`, | ||||
|                                 statusMessage: "Fetch parse error", | ||||
|                                 message: "rest-tools.ts Wrong message model to parse" | ||||
|                             } as RestErrorResponse); | ||||
|                         }); | ||||
|                 } else { | ||||
|                     resolve({ status: response.status, data: response.body }); | ||||
|                 } | ||||
|             } else { | ||||
|                 reject({ | ||||
|                     time: Date().toString(), | ||||
|                     status: response.status, | ||||
|                     error: `${response.body}`, | ||||
|                     statusMessage: "Fetch code error", | ||||
|                     message: "rest-tools.ts Wrong return code" | ||||
|                 } as RestErrorResponse); | ||||
|             } | ||||
|         }).catch((error: any) => { | ||||
|             reject({ | ||||
|                 time: Date(), | ||||
|                 status: 999, | ||||
|                 error: error, | ||||
|                 statusMessage: "Fetch catch error", | ||||
|                 message: "rest-tools.ts detect an error in the fetch request" | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| export function RESTRequestJson<TYPE>(request: RESTRequestType, checker: (data: any) => data is TYPE): Promise<TYPE> { | ||||
|     return new Promise((resolve, reject) => { | ||||
|         RESTRequest(request).then((value: ModelResponseHttp) => { | ||||
|             if (checker(value.data)) { | ||||
|                 resolve(value.data); | ||||
|             } else { | ||||
|                 reject({ | ||||
|                     time: Date().toString(), | ||||
|                     status: 950, | ||||
|                     error: "REST Fail to verify the data", | ||||
|                     statusMessage: "API cast ERROR", | ||||
|                     message: "api.ts Check type as fail" | ||||
|                 } as RestErrorResponse); | ||||
|             } | ||||
|         }).catch((reason: RestErrorResponse) => { | ||||
|             reject(reason); | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
| export function RESTRequestJsonArray<TYPE>(request: RESTRequestType, checker: (data: any) => data is TYPE): Promise<TYPE[]> { | ||||
|     return new Promise((resolve, reject) => { | ||||
|         RESTRequest(request).then((value: ModelResponseHttp) => { | ||||
|             if (isArrayOf(value.data, checker)) { | ||||
|                 resolve(value.data); | ||||
|             } else { | ||||
|                 reject({ | ||||
|                     time: Date().toString(), | ||||
|                     status: 950, | ||||
|                     error: "REST Fail to verify the data", | ||||
|                     statusMessage: "API cast ERROR", | ||||
|                     message: "api.ts Check type as fail" | ||||
|                 } as RestErrorResponse); | ||||
|             } | ||||
|         }).catch((reason: RestErrorResponse) => { | ||||
|             reject(reason); | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| export function RESTRequestVoid(request: RESTRequestType): Promise<void> { | ||||
|     return new Promise((resolve, reject) => { | ||||
|         RESTRequest(request).then((value: ModelResponseHttp) => { | ||||
|             resolve(); | ||||
|         }).catch((reason: RestErrorResponse) => { | ||||
|             reject(reason); | ||||
|         }); | ||||
|     }); | ||||
| } | ||||
| @@ -1,57 +0,0 @@ | ||||
| /** | ||||
|  * API of the server (auto-generated code) | ||||
|  */ | ||||
| import { HTTPMimeType, HTTPRequestModel, ModelResponseHttp, RESTConfig, RESTCallbacks, RESTRequestJson, RESTRequestJsonArray, RESTRequestVoid } from "./rest-tools" | ||||
| import {Long, UserKarideo, UserOut, isUserKarideo, isUserOut, } from "./model" | ||||
| export namespace UserResource { | ||||
|  | ||||
| 	/** | ||||
| 	 * Get a specific user data | ||||
| 	 */ | ||||
| 	export function get({ restConfig, params, }: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			id: Long, | ||||
| 		}, | ||||
| 	}): Promise<UserKarideo> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/users/{id}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}, isUserKarideo); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Get all the users | ||||
| 	 */ | ||||
| 	export function gets({ restConfig, }: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<UserKarideo[]> { | ||||
| 		return RESTRequestJsonArray({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/users", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isUserKarideo); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Get the user personal data | ||||
| 	 */ | ||||
| 	export function getMe({ restConfig, }: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<UserOut> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/users/me", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isUserOut); | ||||
| 	}; | ||||
| } | ||||
| @@ -2,25 +2,75 @@ | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
|  | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="256" | ||||
|    height="256" | ||||
|    viewBox="0 0 67.733333 67.733333" | ||||
|    version="1.1" | ||||
|    id="svg8" | ||||
|    inkscape:version="0.92.4 5da689c313, 2019-01-14" | ||||
|    sodipodi:docname="ikon.svg" | ||||
|    inkscape:version="1.3.2 (091e20ef0f, 2023-11-25, custom)" | ||||
|    sodipodi:docname="ikon_gray.svg" | ||||
|    inkscape:export-filename="/home/heero/dev/perso/appl_pro/NoKomment/plugin/chrome/ikon.png" | ||||
|    inkscape:export-xdpi="7.1250005" | ||||
|    inkscape:export-ydpi="7.1250005"> | ||||
|    inkscape:export-ydpi="7.1250005" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/"> | ||||
|   <defs | ||||
|      id="defs2" /> | ||||
|      id="defs2"> | ||||
|     <filter | ||||
|        style="color-interpolation-filters:sRGB;" | ||||
|        inkscape:label="Drop Shadow" | ||||
|        id="filter5338" | ||||
|        x="-0.12319682" | ||||
|        y="-0.081815216" | ||||
|        width="1.2463936" | ||||
|        height="1.1636304"> | ||||
|       <feFlood | ||||
|          flood-opacity="1" | ||||
|          flood-color="rgb(255,255,255)" | ||||
|          result="flood" | ||||
|          id="feFlood5328" /> | ||||
|       <feComposite | ||||
|          in="flood" | ||||
|          in2="SourceGraphic" | ||||
|          operator="in" | ||||
|          result="composite1" | ||||
|          id="feComposite5330" /> | ||||
|       <feGaussianBlur | ||||
|          in="composite1" | ||||
|          stdDeviation="2.1" | ||||
|          result="blur" | ||||
|          id="feGaussianBlur5332" /> | ||||
|       <feOffset | ||||
|          dx="0" | ||||
|          dy="0" | ||||
|          result="offset" | ||||
|          id="feOffset5334" /> | ||||
|       <feComposite | ||||
|          in="SourceGraphic" | ||||
|          in2="offset" | ||||
|          operator="over" | ||||
|          result="composite2" | ||||
|          id="feComposite5336" /> | ||||
|     </filter> | ||||
|     <filter | ||||
|        inkscape:collect="always" | ||||
|        style="color-interpolation-filters:sRGB" | ||||
|        id="filter1159" | ||||
|        x="-0.11802406" | ||||
|        width="1.2360481" | ||||
|        y="-0.078379973" | ||||
|        height="1.1567599"> | ||||
|       <feGaussianBlur | ||||
|          inkscape:collect="always" | ||||
|          stdDeviation="2.0118255" | ||||
|          id="feGaussianBlur1161" /> | ||||
|     </filter> | ||||
|   </defs> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
| @@ -28,22 +78,31 @@ | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="1.979899" | ||||
|      inkscape:cx="52.480467" | ||||
|      inkscape:cy="138.73493" | ||||
|      inkscape:zoom="7.9195959" | ||||
|      inkscape:cx="100.06824" | ||||
|      inkscape:cy="115.66247" | ||||
|      inkscape:document-units="mm" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="true" | ||||
|      units="px" | ||||
|      inkscape:snap-text-baseline="false" | ||||
|      inkscape:window-width="1918" | ||||
|      inkscape:window-height="1038" | ||||
|      inkscape:window-width="3838" | ||||
|      inkscape:window-height="2118" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="20" | ||||
|      inkscape:window-maximized="1"> | ||||
|      inkscape:window-maximized="1" | ||||
|      inkscape:showpageshadow="2" | ||||
|      inkscape:pagecheckerboard="0" | ||||
|      inkscape:deskcolor="#d1d1d1"> | ||||
|     <inkscape:grid | ||||
|        type="xygrid" | ||||
|        id="grid4504" /> | ||||
|        id="grid4504" | ||||
|        originx="0" | ||||
|        originy="0" | ||||
|        spacingy="1" | ||||
|        spacingx="1" | ||||
|        units="px" | ||||
|        visible="true" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata5"> | ||||
| @@ -61,18 +120,24 @@ | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(0,-229.26668)"> | ||||
|      transform="translate(0,-229.26668)" | ||||
|      style="display:inline"> | ||||
|     <g | ||||
|        aria-label="K" | ||||
|        transform="scale(1.0347881,0.96638145)" | ||||
|        style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:84.55024719px;line-height:1.25;font-family:'DejaVu Sans Mono';-inkscape-font-specification:'DejaVu Sans Mono, Normal';font-variant-ligatures:normal;font-variant-caps:normal;font-variant-numeric:normal;font-feature-settings:normal;text-align:start;letter-spacing:0px;word-spacing:0px;writing-mode:lr-tb;text-anchor:start;fill:#000000;fill-opacity:1;stroke:none;stroke-width:2.11376619" | ||||
|        id="text821"> | ||||
|        id="text821-7" | ||||
|        style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:84.55024719px;line-height:1.25;font-family:'DejaVu Sans Mono';-inkscape-font-specification:'DejaVu Sans Mono, Normal';font-variant-ligatures:normal;font-variant-caps:normal;font-variant-numeric:normal;font-feature-settings:normal;text-align:start;letter-spacing:0px;word-spacing:0px;writing-mode:lr-tb;text-anchor:start;opacity:1;fill:#2b3137;fill-opacity:1;stroke:none;stroke-width:2.11376619;stroke-opacity:1" | ||||
|        transform="matrix(0.8407653,0,0,0.83753055,-37.28971,3.4402954)" | ||||
|        aria-label="K"> | ||||
|       <path | ||||
|          d="m 12.784421,241.62303 h 8.949095 v 27.37877 l 25.568842,-27.37877 6.39221,6.84469 -20.455074,21.90302 20.455074,27.37876 -6.39221,5.47576 -19.176632,-27.37877 -6.39221,6.84469 0,20.53408 h -8.949095 z" | ||||
|          style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:84.55024719px;font-family:'DejaVu Sans Mono';-inkscape-font-specification:'DejaVu Sans Mono, Normal';font-variant-ligatures:normal;font-variant-caps:normal;font-variant-numeric:normal;font-feature-settings:normal;text-align:start;writing-mode:lr-tb;text-anchor:start;stroke-width:2.11376619;fill:#ff0000;fill-opacity:1" | ||||
|          id="path823" | ||||
|          sodipodi:nodetypes="ccccccccccccccccccccccccccccccccccccssccccssscccccccccccccccsscccccsssccccccccccccccssscsscsss" | ||||
|          inkscape:connector-curvature="0" | ||||
|          sodipodi:nodetypes="ccccccccccccc" /> | ||||
|          id="path823-5" | ||||
|          style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:84.5502px;font-family:'DejaVu Sans Mono';-inkscape-font-specification:'DejaVu Sans Mono, Normal';font-variant-ligatures:normal;font-variant-caps:normal;font-variant-numeric:normal;font-feature-settings:normal;text-align:start;writing-mode:lr-tb;text-anchor:start;opacity:0.775;fill:#2b3137;fill-opacity:1;stroke-width:2.11377;filter:url(#filter5338)" | ||||
|          d="m 65.200546,279.9533 h 8.949095 v 27.37877 l 25.568842,-27.37877 6.392207,6.84469 -20.455071,21.90302 20.455071,27.37876 -6.392207,5.47576 -19.176632,-27.37877 -6.39221,6.84469 v 20.53408 h -8.949095 z m 3.913007,39.48974 c -0.26846,-0.43226 -0.592093,-0.92734 -0.887692,-1.37494 l 0.02075,-0.022 c 0.456433,0.27687 0.977308,0.56258 1.422211,0.80755 l 0.407045,0.22999 -0.710959,0.75468 z m 0.591316,3.01367 0.778423,-0.82629 -0.642969,-1.02783 1.022328,-1.08519 1.052938,0.59264 0.80956,-0.85934 -4.631256,-2.27837 -0.913349,0.96952 z m 6.54394,-6.94635 0.762854,-0.80977 -1.289542,-1.22424 0.399591,-0.42416 1.938227,0.53566 0.856265,-0.90892 -2.195068,-0.54992 c 0.187965,-0.54159 0.09714,-1.11827 -0.429654,-1.61839 -0.850547,-0.80747 -1.705543,-0.42955 -2.421693,0.33064 l -1.198771,1.27249 z m -1.168715,-2.64352 -1.004195,-0.95334 0.373643,-0.39662 c 0.40478,-0.42967 0.738271,-0.54092 1.089465,-0.20751 0.351195,0.33341 0.31951,0.73118 -0.08527,1.16085 z m 7.416862,-3.98885 2.345648,-2.48989 -0.68044,-0.64598 -0.788801,0.83731 -2.216914,-2.10465 0.788802,-0.83731 -0.680439,-0.64598 -2.345647,2.48989 0.680439,0.64598 0.788802,-0.83731 2.216913,2.10465 -0.788801,0.83731 z m 6.206624,-6.58829 0.980813,-1.04113 c 0.939292,-0.99705 1.006613,-2.22712 -0.222565,-3.39405 -1.229181,-1.16694 -2.41593,-0.99961 -3.401932,0.047 l -0.934107,0.99155 z m 0.115038,-1.43521 -2.271787,-2.15674 0.124547,-0.13221 c 0.52932,-0.56189 1.150798,-0.69191 2.006834,0.12078 0.856035,0.81268 0.794297,1.47411 0.264954,2.03597 z m 6.226516,-5.29631 2.293753,-2.4348 -0.68044,-0.64599 -1.525707,1.61953 -0.823112,-0.78143 1.250665,-1.32757 -0.674951,-0.64077 -1.250666,1.32757 -0.71885,-0.68245 1.473813,-1.56444 -0.680438,-0.64598 -2.241858,2.37972 z m 7.372646,-7.6936 c 0.80437,-0.85384 0.71213,-2.05798 -0.51156,-3.2197 -1.19626,-1.13568 -2.393561,-1.15578 -3.197932,-0.30194 -0.80437,0.85383 -0.717618,2.05277 0.478638,3.18844 1.223694,1.16173 2.426484,1.18703 3.230854,0.3332 z m -0.6969,-0.66161 c -0.35289,0.37458 -0.97662,0.23116 -1.750342,-0.50339 -0.7408,-0.70328 -0.918234,-1.32045 -0.565349,-1.69503 0.352885,-0.37459 0.976611,-0.23116 1.717411,0.47213 0.77373,0.73454 0.95116,1.3517 0.59828,1.72629 z" /> | ||||
|     </g> | ||||
|     <g | ||||
|        id="text821" | ||||
|        style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:84.55024719px;line-height:1.25;font-family:'DejaVu Sans Mono';-inkscape-font-specification:'DejaVu Sans Mono, Normal';font-variant-ligatures:normal;font-variant-caps:normal;font-variant-numeric:normal;font-feature-settings:normal;text-align:start;letter-spacing:0px;word-spacing:0px;writing-mode:lr-tb;text-anchor:start;fill:#2b3137;fill-opacity:1;stroke:none;stroke-width:2.11376619;stroke-opacity:1;filter:url(#filter1159)" | ||||
|        transform="matrix(1.0347881,0,0,0.96638144,-54.239583,-37.041665)" | ||||
|        aria-label="K" /> | ||||
|   </g> | ||||
| </svg> | ||||
|   | ||||
| Before Width: | Height: | Size: 3.2 KiB After Width: | Height: | Size: 7.5 KiB | 
							
								
								
									
										163
									
								
								front/src/back-api/api/data-resource.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								front/src/back-api/api/data-resource.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { | ||||
| 	HTTPMimeType, | ||||
| 	HTTPRequestModel, | ||||
| 	RESTCallbacks, | ||||
| 	RESTConfig, | ||||
| 	RESTRequestJson, | ||||
| } from "../rest-tools"; | ||||
| import { | ||||
| 	ObjectId, | ||||
| 	isObjectId, | ||||
| } from "../model"; | ||||
|  | ||||
| export namespace DataResource { | ||||
| 	/** | ||||
| 	 * Get back some data from the data environment (with a beautiful name (permit download with basic name) | ||||
| 	 */ | ||||
| 	export function retrieveDataFull({ | ||||
| 			restConfig, | ||||
| 			queries, | ||||
| 			params, | ||||
| 			headers, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		queries: { | ||||
| 			Authorization?: string, | ||||
| 		}, | ||||
| 		params: { | ||||
| 			name: string, | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		headers?: { | ||||
| 			Range?: string, | ||||
| 		}, | ||||
| 	}): Promise<object> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/data/{oid}/{name}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			queries, | ||||
| 			headers, | ||||
| 		}); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Get back some data from the data environment | ||||
| 	 */ | ||||
| 	export function retrieveDataId({ | ||||
| 			restConfig, | ||||
| 			queries, | ||||
| 			params, | ||||
| 			headers, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		queries: { | ||||
| 			Authorization?: string, | ||||
| 		}, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		headers?: { | ||||
| 			Range: string, | ||||
| 		}, | ||||
| 	}): Promise<object> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/data/{oid}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			queries, | ||||
| 			headers, | ||||
| 		}); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Get a thumbnail of from the data environment (if resize is possible) | ||||
| 	 */ | ||||
| 	export function retrieveDataThumbnailId({ | ||||
| 			restConfig, | ||||
| 			queries, | ||||
| 			params, | ||||
| 			headers, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		queries: { | ||||
| 			Authorization?: string, | ||||
| 		}, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		headers?: { | ||||
| 			Range: string, | ||||
| 		}, | ||||
| 	}): Promise<object> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/data/thumbnail/{oid}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			queries, | ||||
| 			headers, | ||||
| 		}); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Upload data in the system | ||||
| 	 */ | ||||
| 	export function uploadMedia({ | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 			callbacks, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		data: { | ||||
| 			file: File, | ||||
| 		}, | ||||
| 		callbacks?: RESTCallbacks, | ||||
| 	}): Promise<ObjectId> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/data/upload", | ||||
| 				requestType: HTTPRequestModel.POST, | ||||
| 				contentType: HTTPMimeType.MULTIPART, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 			callbacks, | ||||
| 		}, isObjectId); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Upload data in the system with an external URI | ||||
| 	 */ | ||||
| 	export function uploadMediaFromUri({ | ||||
| 			restConfig, | ||||
| 			queries, | ||||
| 			callbacks, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		queries: { | ||||
| 			uri?: string, | ||||
| 		}, | ||||
| 		callbacks?: RESTCallbacks, | ||||
| 	}): Promise<ObjectId> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/data/uploadUri", | ||||
| 				requestType: HTTPRequestModel.POST, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			queries, | ||||
| 			callbacks, | ||||
| 		}, isObjectId); | ||||
| 	}; | ||||
| } | ||||
							
								
								
									
										6
									
								
								front/src/back-api/api/front.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								front/src/back-api/api/front.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
|  | ||||
| export namespace Front { | ||||
| } | ||||
							
								
								
									
										33
									
								
								front/src/back-api/api/health-check.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								front/src/back-api/api/health-check.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { | ||||
| 	HTTPMimeType, | ||||
| 	HTTPRequestModel, | ||||
| 	RESTConfig, | ||||
| 	RESTRequestJson, | ||||
| } from "../rest-tools"; | ||||
| import { | ||||
| 	HealthResult, | ||||
| 	isHealthResult, | ||||
| } from "../model"; | ||||
|  | ||||
| export namespace HealthCheck { | ||||
| 	/** | ||||
| 	 * Get the server state (health) | ||||
| 	 */ | ||||
| 	export function getHealth({ | ||||
| 			restConfig, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<HealthResult> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/health_check/", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isHealthResult); | ||||
| 	}; | ||||
| } | ||||
							
								
								
									
										12
									
								
								front/src/back-api/api/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								front/src/back-api/api/index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| export * from "./data-resource" | ||||
| export * from "./front" | ||||
| export * from "./health-check" | ||||
| export * from "./media-resource" | ||||
| export * from "./season-resource" | ||||
| export * from "./series-resource" | ||||
| export * from "./type-resource" | ||||
| export * from "./user-media-advancement-resource" | ||||
| export * from "./user-resource" | ||||
							
								
								
									
										156
									
								
								front/src/back-api/api/media-resource.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								front/src/back-api/api/media-resource.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { | ||||
| 	HTTPMimeType, | ||||
| 	HTTPRequestModel, | ||||
| 	RESTCallbacks, | ||||
| 	RESTConfig, | ||||
| 	RESTRequestJson, | ||||
| 	RESTRequestVoid, | ||||
| } from "../rest-tools"; | ||||
| import { z as zod } from "zod" | ||||
| import { | ||||
| 	Long, | ||||
| 	Media, | ||||
| 	MediaUpdate, | ||||
| 	ObjectId, | ||||
| 	ZodMedia, | ||||
| 	isMedia, | ||||
| } from "../model"; | ||||
|  | ||||
| export namespace MediaResource { | ||||
| 	/** | ||||
| 	 * Get a specific Media with his ID | ||||
| 	 */ | ||||
| 	export function get({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			id: Long, | ||||
| 		}, | ||||
| 	}): Promise<Media> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/media/{id}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}, isMedia); | ||||
| 	}; | ||||
|  | ||||
| 	export const ZodGetsTypeReturn = zod.array(ZodMedia); | ||||
| 	export type GetsTypeReturn = zod.infer<typeof ZodGetsTypeReturn>; | ||||
| 	 | ||||
| 	export function isGetsTypeReturn(data: any): data is GetsTypeReturn { | ||||
| 		try { | ||||
| 			ZodGetsTypeReturn.parse(data); | ||||
| 			return true; | ||||
| 		} catch (e: any) { | ||||
| 			console.log(`Fail to parse data type='ZodGetsTypeReturn' error=${e}`); | ||||
| 			return false; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Get all Media | ||||
| 	 */ | ||||
| 	export function gets({ | ||||
| 			restConfig, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<GetsTypeReturn> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/media/", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isGetsTypeReturn); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Modify a specific Media | ||||
| 	 */ | ||||
| 	export function put({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		data: MediaUpdate, | ||||
| 	}): Promise<Media> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/media/{oid}", | ||||
| 				requestType: HTTPRequestModel.PUT, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}, isMedia); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Remove a specific Media | ||||
| 	 */ | ||||
| 	export function remove({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<void> { | ||||
| 		return RESTRequestVoid({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/media/{oid}", | ||||
| 				requestType: HTTPRequestModel.DELETE, | ||||
| 				contentType: HTTPMimeType.TEXT_PLAIN, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Create a new Media | ||||
| 	 */ | ||||
| 	export function uploadMedia({ | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 			callbacks, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		data: { | ||||
| 			file: File, | ||||
| 			universeId: string, | ||||
| 			season: string, | ||||
| 			typeId: string, | ||||
| 			episode: string, | ||||
| 			title: string, | ||||
| 			seriesId: string, | ||||
| 		}, | ||||
| 		callbacks?: RESTCallbacks, | ||||
| 	}): Promise<Media> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/media/", | ||||
| 				requestType: HTTPRequestModel.POST, | ||||
| 				contentType: HTTPMimeType.MULTIPART, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 			callbacks, | ||||
| 		}, isMedia); | ||||
| 	}; | ||||
| } | ||||
							
								
								
									
										145
									
								
								front/src/back-api/api/season-resource.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								front/src/back-api/api/season-resource.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { | ||||
| 	HTTPMimeType, | ||||
| 	HTTPRequestModel, | ||||
| 	RESTConfig, | ||||
| 	RESTRequestJson, | ||||
| 	RESTRequestVoid, | ||||
| } from "../rest-tools"; | ||||
| import { z as zod } from "zod" | ||||
| import { | ||||
| 	ObjectId, | ||||
| 	Season, | ||||
| 	SeasonCreate, | ||||
| 	SeasonUpdate, | ||||
| 	ZodSeason, | ||||
| 	isSeason, | ||||
| } from "../model"; | ||||
|  | ||||
| export namespace SeasonResource { | ||||
| 	/** | ||||
| 	 * Get all season | ||||
| 	 */ | ||||
| 	export function get({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<Season> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/season/{oid}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}, isSeason); | ||||
| 	}; | ||||
|  | ||||
| 	export const ZodGetsTypeReturn = zod.array(ZodSeason); | ||||
| 	export type GetsTypeReturn = zod.infer<typeof ZodGetsTypeReturn>; | ||||
| 	 | ||||
| 	export function isGetsTypeReturn(data: any): data is GetsTypeReturn { | ||||
| 		try { | ||||
| 			ZodGetsTypeReturn.parse(data); | ||||
| 			return true; | ||||
| 		} catch (e: any) { | ||||
| 			console.log(`Fail to parse data type='ZodGetsTypeReturn' error=${e}`); | ||||
| 			return false; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Get a specific Season with his ID | ||||
| 	 */ | ||||
| 	export function gets({ | ||||
| 			restConfig, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<GetsTypeReturn> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/season/", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isGetsTypeReturn); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Create a new season | ||||
| 	 */ | ||||
| 	export function post({ | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		data: SeasonCreate, | ||||
| 	}): Promise<Season> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/season/", | ||||
| 				requestType: HTTPRequestModel.POST, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 		}, isSeason); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Modify a specific season | ||||
| 	 */ | ||||
| 	export function put({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		data: SeasonUpdate, | ||||
| 	}): Promise<Season> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/season/{oid}", | ||||
| 				requestType: HTTPRequestModel.PUT, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}, isSeason); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Remove a specific season | ||||
| 	 */ | ||||
| 	export function remove({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<void> { | ||||
| 		return RESTRequestVoid({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/season/{oid}", | ||||
| 				requestType: HTTPRequestModel.DELETE, | ||||
| 				contentType: HTTPMimeType.TEXT_PLAIN, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}); | ||||
| 	}; | ||||
| } | ||||
							
								
								
									
										145
									
								
								front/src/back-api/api/series-resource.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								front/src/back-api/api/series-resource.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { | ||||
| 	HTTPMimeType, | ||||
| 	HTTPRequestModel, | ||||
| 	RESTConfig, | ||||
| 	RESTRequestJson, | ||||
| 	RESTRequestVoid, | ||||
| } from "../rest-tools"; | ||||
| import { z as zod } from "zod" | ||||
| import { | ||||
| 	ObjectId, | ||||
| 	Series, | ||||
| 	SeriesCreate, | ||||
| 	SeriesUpdate, | ||||
| 	ZodSeries, | ||||
| 	isSeries, | ||||
| } from "../model"; | ||||
|  | ||||
| export namespace SeriesResource { | ||||
| 	/** | ||||
| 	 * Get a specific Series with his ID | ||||
| 	 */ | ||||
| 	export function get({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<Series> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/series/{oid}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}, isSeries); | ||||
| 	}; | ||||
|  | ||||
| 	export const ZodGetsTypeReturn = zod.array(ZodSeries); | ||||
| 	export type GetsTypeReturn = zod.infer<typeof ZodGetsTypeReturn>; | ||||
| 	 | ||||
| 	export function isGetsTypeReturn(data: any): data is GetsTypeReturn { | ||||
| 		try { | ||||
| 			ZodGetsTypeReturn.parse(data); | ||||
| 			return true; | ||||
| 		} catch (e: any) { | ||||
| 			console.log(`Fail to parse data type='ZodGetsTypeReturn' error=${e}`); | ||||
| 			return false; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Get all Series | ||||
| 	 */ | ||||
| 	export function gets({ | ||||
| 			restConfig, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<GetsTypeReturn> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/series/", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isGetsTypeReturn); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Create a new Series | ||||
| 	 */ | ||||
| 	export function post({ | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		data: SeriesCreate, | ||||
| 	}): Promise<Series> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/series/", | ||||
| 				requestType: HTTPRequestModel.POST, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 		}, isSeries); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Modify a specific Series | ||||
| 	 */ | ||||
| 	export function put({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		data: SeriesUpdate, | ||||
| 	}): Promise<Series> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/series/{oid}", | ||||
| 				requestType: HTTPRequestModel.PUT, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}, isSeries); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Remove a specific Series | ||||
| 	 */ | ||||
| 	export function remove({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<void> { | ||||
| 		return RESTRequestVoid({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/series/{oid}", | ||||
| 				requestType: HTTPRequestModel.DELETE, | ||||
| 				contentType: HTTPMimeType.TEXT_PLAIN, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}); | ||||
| 	}; | ||||
| } | ||||
							
								
								
									
										145
									
								
								front/src/back-api/api/type-resource.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								front/src/back-api/api/type-resource.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| /** | ||||
|  * Interface of the server (auto-generated code) | ||||
|  */ | ||||
| import { | ||||
| 	HTTPMimeType, | ||||
| 	HTTPRequestModel, | ||||
| 	RESTConfig, | ||||
| 	RESTRequestJson, | ||||
| 	RESTRequestVoid, | ||||
| } from "../rest-tools"; | ||||
| import { z as zod } from "zod" | ||||
| import { | ||||
| 	ObjectId, | ||||
| 	Type, | ||||
| 	TypeCreate, | ||||
| 	TypeUpdate, | ||||
| 	ZodType, | ||||
| 	isType, | ||||
| } from "../model"; | ||||
|  | ||||
| export namespace TypeResource { | ||||
| 	/** | ||||
| 	 * Get a specific Type with his ID | ||||
| 	 */ | ||||
| 	export function get({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<Type> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/type/{oid}", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}, isType); | ||||
| 	}; | ||||
|  | ||||
| 	export const ZodGetsTypeReturn = zod.array(ZodType); | ||||
| 	export type GetsTypeReturn = zod.infer<typeof ZodGetsTypeReturn>; | ||||
| 	 | ||||
| 	export function isGetsTypeReturn(data: any): data is GetsTypeReturn { | ||||
| 		try { | ||||
| 			ZodGetsTypeReturn.parse(data); | ||||
| 			return true; | ||||
| 		} catch (e: any) { | ||||
| 			console.log(`Fail to parse data type='ZodGetsTypeReturn' error=${e}`); | ||||
| 			return false; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Get all Type | ||||
| 	 */ | ||||
| 	export function gets({ | ||||
| 			restConfig, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 	}): Promise<GetsTypeReturn> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/type/", | ||||
| 				requestType: HTTPRequestModel.GET, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 		}, isGetsTypeReturn); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Modify a specific Type | ||||
| 	 */ | ||||
| 	export function patch({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 		data: TypeUpdate, | ||||
| 	}): Promise<Type> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/type/{oid}", | ||||
| 				requestType: HTTPRequestModel.PUT, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 			data, | ||||
| 		}, isType); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Create a new Type | ||||
| 	 */ | ||||
| 	export function post({ | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		data: TypeCreate, | ||||
| 	}): Promise<Type> { | ||||
| 		return RESTRequestJson({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/type/", | ||||
| 				requestType: HTTPRequestModel.POST, | ||||
| 				contentType: HTTPMimeType.JSON, | ||||
| 				accept: HTTPMimeType.JSON, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			data, | ||||
| 		}, isType); | ||||
| 	}; | ||||
| 	/** | ||||
| 	 * Remove a specific Type | ||||
| 	 */ | ||||
| 	export function remove({ | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}: { | ||||
| 		restConfig: RESTConfig, | ||||
| 		params: { | ||||
| 			oid: ObjectId, | ||||
| 		}, | ||||
| 	}): Promise<void> { | ||||
| 		return RESTRequestVoid({ | ||||
| 			restModel: { | ||||
| 				endPoint: "/type/{oid}", | ||||
| 				requestType: HTTPRequestModel.DELETE, | ||||
| 				contentType: HTTPMimeType.TEXT_PLAIN, | ||||
| 			}, | ||||
| 			restConfig, | ||||
| 			params, | ||||
| 		}); | ||||
| 	}; | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user