fix: [临时提交]
This commit is contained in:
parent
470d3caf21
commit
6b3a6217b8
|
@ -0,0 +1,455 @@
|
|||
[INFO] Scanning for projects...
|
||||
[WARNING]
|
||||
[WARNING] Some problems were encountered while building the effective model for com.guwan:backend:jar:0.0.1-SNAPSHOT
|
||||
[WARNING] 'dependencies.dependency.(groupId:artifactId:type:classifier)' must be unique: com.aliyun:dysmsapi20170525:jar -> version 2.0.24 vs 3.0.0 @ line 139, column 21
|
||||
[WARNING] 'dependencies.dependency.(groupId:artifactId:type:classifier)' must be unique: commons-io:commons-io:jar -> version 2.11.0 vs 2.18.0 @ line 436, column 21
|
||||
[WARNING]
|
||||
[WARNING] It is highly recommended to fix these problems because they threaten the stability of your build.
|
||||
[WARNING]
|
||||
[WARNING] For this reason, future Maven versions might no longer support building such malformed projects.
|
||||
[WARNING]
|
||||
[INFO]
|
||||
[INFO] -------------------------< com.guwan:backend >--------------------------
|
||||
[INFO] Building backend 0.0.1-SNAPSHOT
|
||||
[INFO] --------------------------------[ jar ]---------------------------------
|
||||
[WARNING] The POM for com.arcsoft.face:arcsoft-sdk-face:jar:4.1.1.0 is missing, no dependency information available
|
||||
[INFO]
|
||||
[INFO] --- maven-dependency-plugin:3.6.1:tree (default-cli) @ backend ---
|
||||
[INFO] com.guwan:backend:jar:0.0.1-SNAPSHOT
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-web:jar:3.2.1:compile
|
||||
[INFO] | +- org.springframework.boot:spring-boot-starter:jar:3.2.1:compile
|
||||
[INFO] | | +- org.springframework.boot:spring-boot:jar:3.2.1:compile
|
||||
[INFO] | | \- jakarta.annotation:jakarta.annotation-api:jar:2.1.1:compile
|
||||
[INFO] | +- org.springframework.boot:spring-boot-starter-json:jar:3.2.1:compile
|
||||
[INFO] | | +- com.fasterxml.jackson.datatype:jackson-datatype-jdk8:jar:2.15.3:compile
|
||||
[INFO] | | \- com.fasterxml.jackson.module:jackson-module-parameter-names:jar:2.15.3:compile
|
||||
[INFO] | +- org.springframework.boot:spring-boot-starter-tomcat:jar:3.2.1:compile
|
||||
[INFO] | | +- org.apache.tomcat.embed:tomcat-embed-core:jar:10.1.17:compile
|
||||
[INFO] | | \- org.apache.tomcat.embed:tomcat-embed-websocket:jar:10.1.17:compile
|
||||
[INFO] | +- org.springframework:spring-web:jar:6.1.2:compile
|
||||
[INFO] | | \- org.springframework:spring-beans:jar:6.1.2:compile
|
||||
[INFO] | \- org.springframework:spring-webmvc:jar:6.1.2:compile
|
||||
[INFO] | \- org.springframework:spring-expression:jar:6.1.2:compile
|
||||
[INFO] +- com.mysql:mysql-connector-j:jar:8.1.0:runtime
|
||||
[INFO] +- org.projectlombok:lombok:jar:1.18.30:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-test:jar:3.2.1:test
|
||||
[INFO] | +- org.springframework.boot:spring-boot-test:jar:3.2.1:test
|
||||
[INFO] | +- org.springframework.boot:spring-boot-test-autoconfigure:jar:3.2.1:test
|
||||
[INFO] | +- com.jayway.jsonpath:json-path:jar:2.8.0:test
|
||||
[INFO] | +- jakarta.xml.bind:jakarta.xml.bind-api:jar:4.0.1:compile
|
||||
[INFO] | | \- jakarta.activation:jakarta.activation-api:jar:2.1.2:compile
|
||||
[INFO] | +- net.minidev:json-smart:jar:2.5.0:test
|
||||
[INFO] | | \- net.minidev:accessors-smart:jar:2.5.0:test
|
||||
[INFO] | | \- org.ow2.asm:asm:jar:9.3:test
|
||||
[INFO] | +- org.assertj:assertj-core:jar:3.24.2:test
|
||||
[INFO] | | \- net.bytebuddy:byte-buddy:jar:1.14.10:test
|
||||
[INFO] | +- org.awaitility:awaitility:jar:4.2.0:test
|
||||
[INFO] | +- org.hamcrest:hamcrest:jar:2.2:test
|
||||
[INFO] | +- org.junit.jupiter:junit-jupiter:jar:5.10.1:test
|
||||
[INFO] | | +- org.junit.jupiter:junit-jupiter-api:jar:5.10.1:test
|
||||
[INFO] | | | +- org.opentest4j:opentest4j:jar:1.3.0:test
|
||||
[INFO] | | | +- org.junit.platform:junit-platform-commons:jar:1.10.1:test
|
||||
[INFO] | | | \- org.apiguardian:apiguardian-api:jar:1.1.2:test
|
||||
[INFO] | | +- org.junit.jupiter:junit-jupiter-params:jar:5.10.1:test
|
||||
[INFO] | | \- org.junit.jupiter:junit-jupiter-engine:jar:5.10.1:test
|
||||
[INFO] | | \- org.junit.platform:junit-platform-engine:jar:1.10.1:test
|
||||
[INFO] | +- org.mockito:mockito-core:jar:5.7.0:test
|
||||
[INFO] | | +- net.bytebuddy:byte-buddy-agent:jar:1.14.10:test
|
||||
[INFO] | | \- org.objenesis:objenesis:jar:3.3:test
|
||||
[INFO] | +- org.mockito:mockito-junit-jupiter:jar:5.7.0:test
|
||||
[INFO] | +- org.skyscreamer:jsonassert:jar:1.5.1:test
|
||||
[INFO] | | \- com.vaadin.external.google:android-json:jar:0.0.20131108.vaadin1:test
|
||||
[INFO] | +- org.springframework:spring-core:jar:6.1.2:compile
|
||||
[INFO] | | \- org.springframework:spring-jcl:jar:6.1.2:compile
|
||||
[INFO] | +- org.springframework:spring-test:jar:6.1.2:test
|
||||
[INFO] | \- org.xmlunit:xmlunit-core:jar:2.9.1:test
|
||||
[INFO] +- com.baomidou:mybatis-plus-boot-starter:jar:3.5.5:compile
|
||||
[INFO] | +- com.baomidou:mybatis-plus:jar:3.5.5:compile
|
||||
[INFO] | | +- com.baomidou:mybatis-plus-core:jar:3.5.5:compile
|
||||
[INFO] | | +- com.baomidou:mybatis-plus-annotation:jar:3.5.5:compile
|
||||
[INFO] | | +- com.baomidou:mybatis-plus-extension:jar:3.5.5:compile
|
||||
[INFO] | | +- org.mybatis:mybatis:jar:3.5.15:compile
|
||||
[INFO] | | \- com.github.jsqlparser:jsqlparser:jar:4.6:compile
|
||||
[INFO] | +- com.baomidou:mybatis-plus-spring-boot-autoconfigure:jar:3.5.5:compile
|
||||
[INFO] | +- org.springframework.boot:spring-boot-autoconfigure:jar:3.2.1:compile
|
||||
[INFO] | \- org.springframework.boot:spring-boot-starter-jdbc:jar:3.2.1:compile
|
||||
[INFO] | +- com.zaxxer:HikariCP:jar:5.0.1:compile
|
||||
[INFO] | \- org.springframework:spring-jdbc:jar:6.1.2:compile
|
||||
[INFO] +- org.mybatis:mybatis-spring:jar:3.0.3:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-data-redis:jar:3.2.1:compile
|
||||
[INFO] | +- io.lettuce:lettuce-core:jar:6.3.0.RELEASE:compile
|
||||
[INFO] | | \- io.projectreactor:reactor-core:jar:3.6.1:compile
|
||||
[INFO] | | \- org.reactivestreams:reactive-streams:jar:1.0.4:compile
|
||||
[INFO] | \- org.springframework.data:spring-data-redis:jar:3.2.1:compile
|
||||
[INFO] | +- org.springframework.data:spring-data-keyvalue:jar:3.2.1:compile
|
||||
[INFO] | \- org.springframework:spring-oxm:jar:6.1.2:compile
|
||||
[INFO] +- org.apache.commons:commons-pool2:jar:2.12.0:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-validation:jar:3.2.1:compile
|
||||
[INFO] | +- org.apache.tomcat.embed:tomcat-embed-el:jar:10.1.17:compile
|
||||
[INFO] | \- org.hibernate.validator:hibernate-validator:jar:8.0.1.Final:compile
|
||||
[INFO] | +- jakarta.validation:jakarta.validation-api:jar:3.0.2:compile
|
||||
[INFO] | +- org.jboss.logging:jboss-logging:jar:3.5.3.Final:compile
|
||||
[INFO] | \- com.fasterxml:classmate:jar:1.6.0:compile
|
||||
[INFO] +- com.fasterxml.jackson.core:jackson-databind:jar:2.15.3:compile
|
||||
[INFO] | +- com.fasterxml.jackson.core:jackson-annotations:jar:2.15.3:compile
|
||||
[INFO] | \- com.fasterxml.jackson.core:jackson-core:jar:2.15.3:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-security:jar:3.2.1:compile
|
||||
[INFO] | +- org.springframework:spring-aop:jar:6.1.2:compile
|
||||
[INFO] | +- org.springframework.security:spring-security-config:jar:6.2.1:compile
|
||||
[INFO] | | \- org.springframework.security:spring-security-core:jar:6.2.1:compile
|
||||
[INFO] | \- org.springframework.security:spring-security-web:jar:6.2.1:compile
|
||||
[INFO] +- io.jsonwebtoken:jjwt-api:jar:0.11.5:compile
|
||||
[INFO] +- io.jsonwebtoken:jjwt-impl:jar:0.11.5:runtime
|
||||
[INFO] +- io.jsonwebtoken:jjwt-jackson:jar:0.11.5:runtime
|
||||
[INFO] +- com.aliyun:dysmsapi20170525:jar:3.0.0:compile
|
||||
[INFO] | +- com.aliyun:tea-util:jar:0.2.22:compile
|
||||
[INFO] | +- com.aliyun:endpoint-util:jar:0.0.7:compile
|
||||
[INFO] | +- com.aliyun:tea:jar:1.2.7:compile
|
||||
[INFO] | | \- org.jacoco:org.jacoco.agent:jar:runtime:0.8.4:compile
|
||||
[INFO] | +- com.aliyun:tea-openapi:jar:0.3.4:compile
|
||||
[INFO] | | +- com.aliyun:credentials-java:jar:0.3.3:compile
|
||||
[INFO] | | | +- org.ini4j:ini4j:jar:0.5.4:compile
|
||||
[INFO] | | | +- com.sun.xml.bind:jaxb-core:jar:4.0.4:compile
|
||||
[INFO] | | | \- com.sun.xml.bind:jaxb-impl:jar:4.0.4:compile
|
||||
[INFO] | | +- com.aliyun:alibabacloud-gateway-spi:jar:0.0.1:compile
|
||||
[INFO] | | \- com.aliyun:tea-xml:jar:0.1.6:compile
|
||||
[INFO] | | \- org.dom4j:dom4j:jar:2.0.3:compile
|
||||
[INFO] | \- com.aliyun:openapiutil:jar:0.2.1:compile
|
||||
[INFO] | \- org.bouncycastle:bcpkix-jdk15on:jar:1.70:compile
|
||||
[INFO] | +- org.bouncycastle:bcprov-jdk15on:jar:1.70:compile
|
||||
[INFO] | \- org.bouncycastle:bcutil-jdk15on:jar:1.70:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-mail:jar:3.2.1:compile
|
||||
[INFO] | +- org.springframework:spring-context-support:jar:6.1.2:compile
|
||||
[INFO] | \- org.eclipse.angus:jakarta.mail:jar:2.0.2:compile
|
||||
[INFO] | \- org.eclipse.angus:angus-activation:jar:2.0.1:runtime
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-thymeleaf:jar:3.2.1:compile
|
||||
[INFO] | \- org.thymeleaf:thymeleaf-spring6:jar:3.1.2.RELEASE:compile
|
||||
[INFO] | \- org.thymeleaf:thymeleaf:jar:3.1.2.RELEASE:compile
|
||||
[INFO] | +- org.attoparser:attoparser:jar:2.0.7.RELEASE:compile
|
||||
[INFO] | \- org.unbescape:unbescape:jar:1.1.6.RELEASE:compile
|
||||
[INFO] +- org.apache.commons:commons-lang3:jar:3.13.0:compile
|
||||
[INFO] +- cn.hutool:hutool-all:jar:5.8.18:compile
|
||||
[INFO] +- com.fasterxml.jackson.datatype:jackson-datatype-jsr310:jar:2.15.3:compile
|
||||
[INFO] +- com.alibaba:fastjson:jar:2.0.53:compile
|
||||
[INFO] | \- com.alibaba.fastjson2:fastjson2-extension:jar:2.0.53:compile
|
||||
[INFO] | \- com.alibaba.fastjson2:fastjson2:jar:2.0.53:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-aop:jar:3.2.1:compile
|
||||
[INFO] +- org.aspectj:aspectjweaver:jar:1.9.21:compile
|
||||
[INFO] +- io.minio:minio:jar:8.5.7:compile
|
||||
[INFO] | +- com.carrotsearch.thirdparty:simple-xml-safe:jar:2.7.1:compile
|
||||
[INFO] | +- com.google.guava:guava:jar:32.1.3-jre:compile
|
||||
[INFO] | | +- com.google.guava:failureaccess:jar:1.0.1:compile
|
||||
[INFO] | | +- com.google.guava:listenablefuture:jar:9999.0-empty-to-avoid-conflict-with-guava:compile
|
||||
[INFO] | | \- com.google.j2objc:j2objc-annotations:jar:2.8:compile
|
||||
[INFO] | +- org.bouncycastle:bcprov-jdk18on:jar:1.76:compile
|
||||
[INFO] | +- org.apache.commons:commons-compress:jar:1.24.0:compile
|
||||
[INFO] | \- org.xerial.snappy:snappy-java:jar:1.1.10.5:compile
|
||||
[INFO] +- commons-io:commons-io:jar:2.18.0:compile
|
||||
[INFO] +- ai.djl:api:jar:0.25.0:compile
|
||||
[INFO] | +- com.google.code.gson:gson:jar:2.10.1:compile
|
||||
[INFO] | +- net.java.dev.jna:jna:jar:5.13.0:compile
|
||||
[INFO] | \- org.slf4j:slf4j-api:jar:2.0.9:compile
|
||||
[INFO] +- ai.djl.pytorch:pytorch-engine:jar:0.25.0:compile
|
||||
[INFO] +- ai.djl.pytorch:pytorch-model-zoo:jar:0.25.0:compile
|
||||
[INFO] +- org.apache.hadoop:hadoop-client:jar:3.3.6:compile
|
||||
[INFO] | +- org.apache.hadoop:hadoop-common:jar:3.3.6:compile
|
||||
[INFO] | | +- org.apache.hadoop.thirdparty:hadoop-shaded-protobuf_3_7:jar:1.1.1:compile
|
||||
[INFO] | | +- commons-net:commons-net:jar:3.9.0:compile
|
||||
[INFO] | | +- commons-collections:commons-collections:jar:3.2.2:compile
|
||||
[INFO] | | +- org.eclipse.jetty:jetty-servlet:jar:9.4.51.v20230217:compile
|
||||
[INFO] | | | \- org.eclipse.jetty:jetty-security:jar:12.0.5:compile
|
||||
[INFO] | | +- org.eclipse.jetty:jetty-webapp:jar:9.4.51.v20230217:compile
|
||||
[INFO] | | | \- org.eclipse.jetty:jetty-xml:jar:12.0.5:compile
|
||||
[INFO] | | +- javax.servlet.jsp:jsp-api:jar:2.1:runtime
|
||||
[INFO] | | +- com.sun.jersey:jersey-servlet:jar:1.19.4:compile
|
||||
[INFO] | | +- org.apache.commons:commons-configuration2:jar:2.8.0:compile
|
||||
[INFO] | | +- org.apache.commons:commons-text:jar:1.10.0:compile
|
||||
[INFO] | | +- org.apache.avro:avro:jar:1.7.7:compile
|
||||
[INFO] | | | +- org.codehaus.jackson:jackson-core-asl:jar:1.9.13:compile
|
||||
[INFO] | | | +- org.codehaus.jackson:jackson-mapper-asl:jar:1.9.13:compile
|
||||
[INFO] | | | \- com.thoughtworks.paranamer:paranamer:jar:2.3:compile
|
||||
[INFO] | | +- com.google.re2j:re2j:jar:1.1:compile
|
||||
[INFO] | | +- org.apache.hadoop:hadoop-auth:jar:3.3.6:compile
|
||||
[INFO] | | | +- com.nimbusds:nimbus-jose-jwt:jar:9.8.1:compile
|
||||
[INFO] | | | | \- com.github.stephenc.jcip:jcip-annotations:jar:1.0-1:compile
|
||||
[INFO] | | | +- org.apache.curator:curator-framework:jar:5.2.0:compile
|
||||
[INFO] | | | \- org.apache.kerby:kerb-simplekdc:jar:1.0.1:compile
|
||||
[INFO] | | | +- org.apache.kerby:kerb-client:jar:1.0.1:compile
|
||||
[INFO] | | | | +- org.apache.kerby:kerby-config:jar:1.0.1:compile
|
||||
[INFO] | | | | +- org.apache.kerby:kerb-common:jar:1.0.1:compile
|
||||
[INFO] | | | | | \- org.apache.kerby:kerb-crypto:jar:1.0.1:compile
|
||||
[INFO] | | | | +- org.apache.kerby:kerb-util:jar:1.0.1:compile
|
||||
[INFO] | | | | \- org.apache.kerby:token-provider:jar:1.0.1:compile
|
||||
[INFO] | | | \- org.apache.kerby:kerb-admin:jar:1.0.1:compile
|
||||
[INFO] | | | +- org.apache.kerby:kerb-server:jar:1.0.1:compile
|
||||
[INFO] | | | | \- org.apache.kerby:kerb-identity:jar:1.0.1:compile
|
||||
[INFO] | | | \- org.apache.kerby:kerby-xdr:jar:1.0.1:compile
|
||||
[INFO] | | +- org.apache.curator:curator-client:jar:5.2.0:compile
|
||||
[INFO] | | +- org.apache.curator:curator-recipes:jar:5.2.0:compile
|
||||
[INFO] | | +- io.dropwizard.metrics:metrics-core:jar:4.2.23:compile
|
||||
[INFO] | | +- org.apache.kerby:kerb-core:jar:1.0.1:compile
|
||||
[INFO] | | | \- org.apache.kerby:kerby-pkix:jar:1.0.1:compile
|
||||
[INFO] | | | +- org.apache.kerby:kerby-asn1:jar:1.0.1:compile
|
||||
[INFO] | | | \- org.apache.kerby:kerby-util:jar:1.0.1:compile
|
||||
[INFO] | | +- org.codehaus.woodstox:stax2-api:jar:4.2.1:compile
|
||||
[INFO] | | +- com.fasterxml.woodstox:woodstox-core:jar:5.4.0:compile
|
||||
[INFO] | | \- dnsjava:dnsjava:jar:2.1.7:compile
|
||||
[INFO] | +- org.apache.hadoop:hadoop-hdfs-client:jar:3.3.6:compile
|
||||
[INFO] | +- org.apache.hadoop:hadoop-yarn-api:jar:3.3.6:compile
|
||||
[INFO] | | \- javax.xml.bind:jaxb-api:jar:2.2.11:compile
|
||||
[INFO] | +- org.apache.hadoop:hadoop-yarn-client:jar:3.3.6:compile
|
||||
[INFO] | | +- org.eclipse.jetty.websocket:websocket-client:jar:9.4.51.v20230217:compile
|
||||
[INFO] | | | +- org.eclipse.jetty:jetty-client:jar:12.0.5:compile
|
||||
[INFO] | | | | \- org.eclipse.jetty:jetty-alpn-client:jar:12.0.5:compile
|
||||
[INFO] | | | \- org.eclipse.jetty.websocket:websocket-common:jar:9.4.51.v20230217:compile
|
||||
[INFO] | | | \- org.eclipse.jetty.websocket:websocket-api:jar:9.4.51.v20230217:compile
|
||||
[INFO] | | \- org.jline:jline:jar:3.9.0:compile
|
||||
[INFO] | +- org.apache.hadoop:hadoop-mapreduce-client-core:jar:3.3.6:compile
|
||||
[INFO] | | \- org.apache.hadoop:hadoop-yarn-common:jar:3.3.6:compile
|
||||
[INFO] | | +- com.sun.jersey:jersey-client:jar:1.19.4:compile
|
||||
[INFO] | | \- com.fasterxml.jackson.module:jackson-module-jaxb-annotations:jar:2.15.3:compile
|
||||
[INFO] | +- org.apache.hadoop:hadoop-mapreduce-client-jobclient:jar:3.3.6:compile
|
||||
[INFO] | | \- org.apache.hadoop:hadoop-mapreduce-client-common:jar:3.3.6:compile
|
||||
[INFO] | \- org.apache.hadoop:hadoop-annotations:jar:3.3.6:compile
|
||||
[INFO] +- org.apache.hadoop:hadoop-hdfs:jar:3.3.6:compile
|
||||
[INFO] | +- org.apache.hadoop.thirdparty:hadoop-shaded-guava:jar:1.1.1:compile
|
||||
[INFO] | +- org.eclipse.jetty:jetty-server:jar:12.0.5:compile
|
||||
[INFO] | | +- org.eclipse.jetty:jetty-http:jar:12.0.5:compile
|
||||
[INFO] | | \- org.eclipse.jetty:jetty-io:jar:12.0.5:compile
|
||||
[INFO] | +- org.eclipse.jetty:jetty-util:jar:12.0.5:compile
|
||||
[INFO] | +- org.eclipse.jetty:jetty-util-ajax:jar:12.0.5:compile
|
||||
[INFO] | +- com.sun.jersey:jersey-core:jar:1.19.4:compile
|
||||
[INFO] | | \- javax.ws.rs:jsr311-api:jar:1.1.1:compile
|
||||
[INFO] | +- com.sun.jersey:jersey-server:jar:1.19.4:compile
|
||||
[INFO] | +- commons-cli:commons-cli:jar:1.2:compile
|
||||
[INFO] | +- commons-codec:commons-codec:jar:1.16.0:compile
|
||||
[INFO] | +- commons-logging:commons-logging:jar:1.1.3:compile
|
||||
[INFO] | +- commons-daemon:commons-daemon:jar:1.0.13:compile
|
||||
[INFO] | +- ch.qos.reload4j:reload4j:jar:1.2.22:compile
|
||||
[INFO] | +- com.google.protobuf:protobuf-java:jar:2.5.0:compile
|
||||
[INFO] | +- javax.servlet:javax.servlet-api:jar:3.1.0:compile
|
||||
[INFO] | +- io.netty:netty:jar:3.10.6.Final:compile
|
||||
[INFO] | \- org.fusesource.leveldbjni:leveldbjni-all:jar:1.8:compile
|
||||
[INFO] +- org.springdoc:springdoc-openapi-starter-webmvc-ui:jar:2.3.0:compile
|
||||
[INFO] | +- org.springdoc:springdoc-openapi-starter-webmvc-api:jar:2.3.0:compile
|
||||
[INFO] | | \- org.springdoc:springdoc-openapi-starter-common:jar:2.3.0:compile
|
||||
[INFO] | | \- io.swagger.core.v3:swagger-core-jakarta:jar:2.2.19:compile
|
||||
[INFO] | | +- io.swagger.core.v3:swagger-annotations-jakarta:jar:2.2.19:compile
|
||||
[INFO] | | \- io.swagger.core.v3:swagger-models-jakarta:jar:2.2.19:compile
|
||||
[INFO] | \- org.webjars:swagger-ui:jar:5.10.3:compile
|
||||
[INFO] +- com.arcsoft.face:arcsoft-sdk-face:jar:4.1.1.0:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-websocket:jar:3.2.1:compile
|
||||
[INFO] | +- org.springframework:spring-messaging:jar:6.1.2:compile
|
||||
[INFO] | \- org.springframework:spring-websocket:jar:6.1.2:compile
|
||||
[INFO] +- io.netty:netty-all:jar:4.1.94.Final:compile
|
||||
[INFO] | +- io.netty:netty-buffer:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-dns:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-haproxy:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-http:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-http2:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-memcache:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-mqtt:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-redis:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-smtp:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-socks:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-stomp:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-codec-xml:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-common:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-handler:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-native-unix-common:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-handler-proxy:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-handler-ssl-ocsp:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-resolver:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-resolver-dns:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-rxtx:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-sctp:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-udt:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-classes-epoll:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-classes-kqueue:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-resolver-dns-classes-macos:jar:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-native-epoll:jar:linux-x86_64:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-native-epoll:jar:linux-aarch_64:4.1.104.Final:runtime
|
||||
[INFO] | +- io.netty:netty-transport-native-kqueue:jar:osx-x86_64:4.1.104.Final:compile
|
||||
[INFO] | +- io.netty:netty-transport-native-kqueue:jar:osx-aarch_64:4.1.104.Final:runtime
|
||||
[INFO] | +- io.netty:netty-resolver-dns-native-macos:jar:osx-x86_64:4.1.104.Final:runtime
|
||||
[INFO] | \- io.netty:netty-resolver-dns-native-macos:jar:osx-aarch_64:4.1.104.Final:runtime
|
||||
[INFO] +- cn.easy-es:easy-es-boot-starter:jar:1.1.1:compile
|
||||
[INFO] | \- cn.easy-es:easy-es-core:jar:1.1.1:compile
|
||||
[INFO] | \- cn.easy-es:easy-es-extension:jar:1.1.1:compile
|
||||
[INFO] | +- cn.easy-es:easy-es-annotation:jar:1.1.1:compile
|
||||
[INFO] | \- cn.easy-es:easy-es-common:jar:1.1.1:compile
|
||||
[INFO] +- org.elasticsearch.client:elasticsearch-rest-high-level-client:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch.plugin:mapper-extras-client:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch.plugin:parent-join-client:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch.plugin:aggs-matrix-stats-client:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch.plugin:rank-eval-client:jar:7.14.0:compile
|
||||
[INFO] | \- org.elasticsearch.plugin:lang-mustache-client:jar:7.14.0:compile
|
||||
[INFO] | \- com.github.spullara.mustache.java:compiler:jar:0.9.6:compile
|
||||
[INFO] +- org.elasticsearch:elasticsearch:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch:elasticsearch-core:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch:elasticsearch-secure-sm:jar:7.14.0:compile
|
||||
[INFO] | +- org.elasticsearch:elasticsearch-x-content:jar:7.14.0:compile
|
||||
[INFO] | | +- com.fasterxml.jackson.dataformat:jackson-dataformat-smile:jar:2.15.3:compile
|
||||
[INFO] | | +- com.fasterxml.jackson.dataformat:jackson-dataformat-yaml:jar:2.15.3:compile
|
||||
[INFO] | | \- com.fasterxml.jackson.dataformat:jackson-dataformat-cbor:jar:2.15.3:compile
|
||||
[INFO] | +- org.elasticsearch:elasticsearch-geo:jar:7.14.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-core:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-analyzers-common:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-backward-codecs:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-grouping:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-highlighter:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-join:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-memory:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-misc:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-queries:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-queryparser:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-sandbox:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-spatial-extras:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-spatial3d:jar:8.9.0:compile
|
||||
[INFO] | +- org.apache.lucene:lucene-suggest:jar:8.9.0:compile
|
||||
[INFO] | +- org.elasticsearch:elasticsearch-cli:jar:7.14.0:compile
|
||||
[INFO] | | \- net.sf.jopt-simple:jopt-simple:jar:5.0.2:compile
|
||||
[INFO] | +- com.carrotsearch:hppc:jar:0.8.1:compile
|
||||
[INFO] | +- org.lz4:lz4-java:jar:1.8.0:compile
|
||||
[INFO] | +- joda-time:joda-time:jar:2.10.10:compile
|
||||
[INFO] | +- com.tdunning:t-digest:jar:3.2:compile
|
||||
[INFO] | +- org.hdrhistogram:HdrHistogram:jar:2.1.9:compile
|
||||
[INFO] | +- org.elasticsearch:jna:jar:5.7.0-1:compile
|
||||
[INFO] | \- org.elasticsearch:elasticsearch-plugin-classloader:jar:7.14.0:runtime
|
||||
[INFO] +- org.elasticsearch.client:elasticsearch-rest-client:jar:7.14.0:compile
|
||||
[INFO] | +- org.apache.httpcomponents:httpclient:jar:4.5.10:compile
|
||||
[INFO] | +- org.apache.httpcomponents:httpcore:jar:4.4.16:compile
|
||||
[INFO] | +- org.apache.httpcomponents:httpasyncclient:jar:4.1.5:compile
|
||||
[INFO] | \- org.apache.httpcomponents:httpcore-nio:jar:4.4.16:compile
|
||||
[INFO] +- com.squareup.okhttp3:okhttp:jar:4.12.0:compile
|
||||
[INFO] | +- com.squareup.okio:okio:jar:3.6.0:compile
|
||||
[INFO] | | \- com.squareup.okio:okio-jvm:jar:3.6.0:compile
|
||||
[INFO] | | \- org.jetbrains.kotlin:kotlin-stdlib-common:jar:1.9.21:compile
|
||||
[INFO] | \- org.jetbrains.kotlin:kotlin-stdlib-jdk8:jar:1.9.21:compile
|
||||
[INFO] | +- org.jetbrains.kotlin:kotlin-stdlib:jar:1.9.21:compile
|
||||
[INFO] | | \- org.jetbrains:annotations:jar:13.0:compile
|
||||
[INFO] | \- org.jetbrains.kotlin:kotlin-stdlib-jdk7:jar:1.9.21:compile
|
||||
[INFO] +- com.mongoplus:mongo-plus-boot-starter:jar:2.1.6.1:compile
|
||||
[INFO] | +- com.mongoplus:mongo-plus-core:jar:2.1.6:compile
|
||||
[INFO] | | \- com.mongoplus:mongo-plus-annotation:jar:2.1.6:compile
|
||||
[INFO] | \- org.springframework:spring-tx:jar:6.1.2:compile
|
||||
[INFO] +- com.mongoplus:mongo-plus-solon-plugin:jar:2.1.6.1:compile
|
||||
[INFO] +- org.springframework.boot:spring-boot-starter-data-mongodb:jar:3.2.1:compile
|
||||
[INFO] | +- org.mongodb:mongodb-driver-sync:jar:4.11.1:compile
|
||||
[INFO] | | +- org.mongodb:bson:jar:4.11.1:compile
|
||||
[INFO] | | \- org.mongodb:mongodb-driver-core:jar:4.11.1:compile
|
||||
[INFO] | | \- org.mongodb:bson-record-codec:jar:4.11.1:runtime
|
||||
[INFO] | \- org.springframework.data:spring-data-mongodb:jar:4.2.1:compile
|
||||
[INFO] | \- org.springframework.data:spring-data-commons:jar:3.2.1:compile
|
||||
[INFO] +- org.springframework.kafka:spring-kafka:jar:3.0.9:compile
|
||||
[INFO] | +- org.springframework:spring-context:jar:6.1.2:compile
|
||||
[INFO] | +- org.springframework.retry:spring-retry:jar:2.0.5:compile
|
||||
[INFO] | +- org.apache.kafka:kafka-clients:jar:3.6.1:compile
|
||||
[INFO] | | \- com.github.luben:zstd-jni:jar:1.5.5-1:runtime
|
||||
[INFO] | +- io.micrometer:micrometer-observation:jar:1.12.1:compile
|
||||
[INFO] | | \- io.micrometer:micrometer-commons:jar:1.12.1:compile
|
||||
[INFO] | \- com.google.code.findbugs:jsr305:jar:3.0.2:compile
|
||||
[INFO] +- org.codehaus.janino:janino:jar:3.1.11:compile
|
||||
[INFO] | \- org.codehaus.janino:commons-compiler:jar:3.1.11:compile
|
||||
[INFO] +- com.github.ben-manes.caffeine:caffeine:jar:3.1.8:compile
|
||||
[INFO] | +- org.checkerframework:checker-qual:jar:3.37.0:compile
|
||||
[INFO] | \- com.google.errorprone:error_prone_annotations:jar:2.21.1:compile
|
||||
[INFO] +- net.logstash.logback:logstash-logback-encoder:jar:6.6:compile
|
||||
[INFO] +- org.springframework.cloud:spring-cloud-starter-openfeign:jar:4.1.2:compile
|
||||
[INFO] | +- org.springframework.cloud:spring-cloud-starter:jar:4.1.3:compile
|
||||
[INFO] | | +- org.springframework.cloud:spring-cloud-context:jar:4.1.3:compile
|
||||
[INFO] | | \- org.springframework.security:spring-security-rsa:jar:1.1.3:compile
|
||||
[INFO] | +- org.springframework.cloud:spring-cloud-openfeign-core:jar:4.1.2:compile
|
||||
[INFO] | | +- io.github.openfeign.form:feign-form-spring:jar:3.8.0:compile
|
||||
[INFO] | | | \- io.github.openfeign.form:feign-form:jar:3.8.0:compile
|
||||
[INFO] | | \- commons-fileupload:commons-fileupload:jar:1.5:compile
|
||||
[INFO] | +- org.springframework.cloud:spring-cloud-commons:jar:4.1.3:compile
|
||||
[INFO] | | \- org.springframework.security:spring-security-crypto:jar:6.2.1:compile
|
||||
[INFO] | +- io.github.openfeign:feign-core:jar:13.2.1:compile
|
||||
[INFO] | \- io.github.openfeign:feign-slf4j:jar:13.2.1:compile
|
||||
[INFO] +- com.xuxueli:xxl-job-core:jar:2.4.0:compile
|
||||
[INFO] | \- org.apache.groovy:groovy:jar:4.0.16:compile
|
||||
[INFO] +- com.github.docker-java:docker-java:jar:3.2.13:compile
|
||||
[INFO] | +- com.github.docker-java:docker-java-core:jar:3.2.13:compile
|
||||
[INFO] | | \- com.github.docker-java:docker-java-api:jar:3.2.13:compile
|
||||
[INFO] | +- com.github.docker-java:docker-java-transport-jersey:jar:3.2.13:compile
|
||||
[INFO] | | +- com.fasterxml.jackson.jaxrs:jackson-jaxrs-json-provider:jar:2.15.3:compile
|
||||
[INFO] | | | \- com.fasterxml.jackson.jaxrs:jackson-jaxrs-base:jar:2.15.3:compile
|
||||
[INFO] | | +- org.glassfish.jersey.connectors:jersey-apache-connector:jar:3.1.5:compile
|
||||
[INFO] | | | +- org.glassfish.jersey.core:jersey-common:jar:3.1.5:compile
|
||||
[INFO] | | | | \- org.glassfish.hk2:osgi-resource-locator:jar:1.0.3:compile
|
||||
[INFO] | | | \- jakarta.ws.rs:jakarta.ws.rs-api:jar:3.1.0:compile
|
||||
[INFO] | | +- org.glassfish.jersey.core:jersey-client:jar:3.1.5:compile
|
||||
[INFO] | | | \- jakarta.inject:jakarta.inject-api:jar:2.0.1:compile
|
||||
[INFO] | | +- org.glassfish.jersey.inject:jersey-hk2:jar:3.1.5:compile
|
||||
[INFO] | | | \- org.glassfish.hk2:hk2-locator:jar:3.0.5:compile
|
||||
[INFO] | | | +- org.glassfish.hk2.external:aopalliance-repackaged:jar:3.0.5:compile
|
||||
[INFO] | | | +- org.glassfish.hk2:hk2-api:jar:3.0.5:compile
|
||||
[INFO] | | | \- org.glassfish.hk2:hk2-utils:jar:3.0.5:compile
|
||||
[INFO] | | +- com.kohlschutter.junixsocket:junixsocket-common:jar:2.3.2:compile
|
||||
[INFO] | | \- com.kohlschutter.junixsocket:junixsocket-native-common:jar:2.3.2:compile
|
||||
[INFO] | +- com.github.docker-java:docker-java-transport-netty:jar:3.2.13:compile
|
||||
[INFO] | \- org.slf4j:jcl-over-slf4j:jar:2.0.9:compile
|
||||
[INFO] +- com.github.docker-java:docker-java-transport-httpclient5:jar:3.2.13:compile
|
||||
[INFO] | \- com.github.docker-java:docker-java-transport:jar:3.2.13:compile
|
||||
[INFO] +- org.apache.httpcomponents.client5:httpclient5:jar:5.2:compile
|
||||
[INFO] | \- org.apache.httpcomponents.core5:httpcore5-h2:jar:5.2.4:compile
|
||||
[INFO] +- org.apache.httpcomponents.core5:httpcore5:jar:5.2:compile
|
||||
[INFO] +- org.reflections:reflections:jar:0.10.2:compile
|
||||
[INFO] | \- org.javassist:javassist:jar:3.28.0-GA:compile
|
||||
[INFO] +- io.swagger.parser.v3:swagger-parser:jar:2.1.25:compile
|
||||
[INFO] | +- io.swagger.parser.v3:swagger-parser-v2-converter:jar:2.1.25:compile
|
||||
[INFO] | | +- io.swagger:swagger-core:jar:1.6.15:compile
|
||||
[INFO] | | | \- io.swagger:swagger-models:jar:1.6.15:compile
|
||||
[INFO] | | | \- io.swagger:swagger-annotations:jar:1.6.15:compile
|
||||
[INFO] | | +- io.swagger:swagger-parser:jar:1.0.73:compile
|
||||
[INFO] | | | \- io.swagger:swagger-parser-safe-url-resolver:jar:1.0.73:compile
|
||||
[INFO] | | +- io.swagger:swagger-compat-spec-parser:jar:1.0.73:compile
|
||||
[INFO] | | | +- com.github.java-json-tools:json-schema-validator:jar:2.2.14:compile
|
||||
[INFO] | | | | +- com.github.java-json-tools:jackson-coreutils-equivalence:jar:1.0:compile
|
||||
[INFO] | | | | +- com.github.java-json-tools:json-schema-core:jar:1.2.14:compile
|
||||
[INFO] | | | | | +- com.github.java-json-tools:uri-template:jar:0.10:compile
|
||||
[INFO] | | | | | \- org.mozilla:rhino:jar:1.7.7.2:compile
|
||||
[INFO] | | | | \- com.googlecode.libphonenumber:libphonenumber:jar:8.11.1:compile
|
||||
[INFO] | | | \- com.github.java-json-tools:json-patch:jar:1.13:compile
|
||||
[INFO] | | | +- com.github.java-json-tools:msg-simple:jar:1.2:compile
|
||||
[INFO] | | | | \- com.github.java-json-tools:btf:jar:1.3:compile
|
||||
[INFO] | | | \- com.github.java-json-tools:jackson-coreutils:jar:2.0:compile
|
||||
[INFO] | | +- io.swagger.core.v3:swagger-models:jar:2.2.28:compile
|
||||
[INFO] | | \- io.swagger.parser.v3:swagger-parser-core:jar:2.1.25:compile
|
||||
[INFO] | +- io.swagger.parser.v3:swagger-parser-v3:jar:2.1.25:compile
|
||||
[INFO] | | +- io.swagger.core.v3:swagger-core:jar:2.2.28:compile
|
||||
[INFO] | | | \- io.swagger.core.v3:swagger-annotations:jar:2.2.28:compile
|
||||
[INFO] | | \- io.swagger.parser.v3:swagger-parser-safe-url-resolver:jar:2.1.25:compile
|
||||
[INFO] | \- org.yaml:snakeyaml:jar:2.2:compile
|
||||
[INFO] +- org.apache.poi:poi-ooxml:jar:5.2.3:compile
|
||||
[INFO] | +- org.apache.poi:poi:jar:5.2.3:compile
|
||||
[INFO] | | \- com.zaxxer:SparseBitSet:jar:1.2:compile
|
||||
[INFO] | +- org.apache.poi:poi-ooxml-lite:jar:5.2.3:compile
|
||||
[INFO] | +- org.apache.xmlbeans:xmlbeans:jar:5.1.1:compile
|
||||
[INFO] | | \- xml-apis:xml-apis:jar:1.4.01:compile
|
||||
[INFO] | +- com.github.virtuald:curvesapi:jar:1.07:compile
|
||||
[INFO] | \- org.apache.commons:commons-collections4:jar:4.4:compile
|
||||
[INFO] +- org.apache.poi:poi-scratchpad:jar:5.2.3:compile
|
||||
[INFO] | \- org.apache.commons:commons-math3:jar:3.6.1:compile
|
||||
[INFO] +- net.sf.dozer:dozer:jar:5.5.1:compile
|
||||
[INFO] | \- commons-beanutils:commons-beanutils:jar:1.9.1:compile
|
||||
[INFO] \- dev.langchain4j:langchain4j-community-dashscope-spring-boot-starter:jar:1.0.0-beta2:compile
|
||||
[INFO] +- dev.langchain4j:langchain4j-community-dashscope:jar:1.0.0-beta2:compile
|
||||
[INFO] | +- dev.langchain4j:langchain4j-core:jar:1.0.0-beta2:compile
|
||||
[INFO] | | \- org.jspecify:jspecify:jar:1.0.0:compile
|
||||
[INFO] | \- com.alibaba:dashscope-sdk-java:jar:2.18.3:compile
|
||||
[INFO] | +- io.reactivex.rxjava2:rxjava:jar:2.2.21:compile
|
||||
[INFO] | +- com.squareup.okhttp3:logging-interceptor:jar:4.12.0:compile
|
||||
[INFO] | +- com.squareup.okhttp3:okhttp-sse:jar:4.12.0:compile
|
||||
[INFO] | \- com.github.victools:jsonschema-generator:jar:4.31.1:compile
|
||||
[INFO] \- ch.qos.logback:logback-classic:jar:1.4.14:compile
|
||||
[INFO] \- ch.qos.logback:logback-core:jar:1.4.14:compile
|
||||
[INFO] ------------------------------------------------------------------------
|
||||
[INFO] BUILD SUCCESS
|
||||
[INFO] ------------------------------------------------------------------------
|
||||
[INFO] Total time: 2.098 s
|
||||
[INFO] Finished at: 2025-03-20T23:57:43+08:00
|
||||
[INFO] ------------------------------------------------------------------------
|
|
@ -0,0 +1,6 @@
|
|||
[2025-03-20 23:38:51.250] - 28460 信息 [main] --- org.apache.catalina.core.StandardService: Starting service [Tomcat]
|
||||
[2025-03-20 23:38:51.258] - 28460 信息 [main] --- org.apache.catalina.core.StandardEngine: Starting Servlet engine: [Apache Tomcat/10.1.17]
|
||||
[2025-03-20 23:38:51.347] - 28460 信息 [main] --- org.apache.catalina.core.ContainerBase.[Tomcat].[localhost].[/]: Initializing Spring embedded WebApplicationContext
|
||||
[2025-03-20 23:38:54.534] - 28460 信息 [main] --- easy-es: ===> manual index mode activated
|
||||
[2025-03-20 23:38:54.624] - 28460 信息 [main] --- easy-es: ===> manual index mode activated
|
||||
[2025-03-20 23:39:03.199] - 28460 信息 [main] --- easy-es: ===> manual index mode activated
|
|
@ -0,0 +1,6 @@
|
|||
[2025-03-20 23:37:40.090] - 23732 信息 [main] --- org.apache.catalina.core.StandardService: Starting service [Tomcat]
|
||||
[2025-03-20 23:37:40.098] - 23732 信息 [main] --- org.apache.catalina.core.StandardEngine: Starting Servlet engine: [Apache Tomcat/10.1.17]
|
||||
[2025-03-20 23:37:40.196] - 23732 信息 [main] --- org.apache.catalina.core.ContainerBase.[Tomcat].[localhost].[/]: Initializing Spring embedded WebApplicationContext
|
||||
[2025-03-20 23:37:44.418] - 23732 信息 [main] --- easy-es: ===> manual index mode activated
|
||||
[2025-03-20 23:37:44.511] - 23732 信息 [main] --- easy-es: ===> manual index mode activated
|
||||
[2025-03-20 23:37:53.289] - 23732 信息 [main] --- easy-es: ===> manual index mode activated
|
122
pom.xml
122
pom.xml
|
@ -29,6 +29,8 @@
|
|||
<properties>
|
||||
<java.version>21</java.version>
|
||||
<mybatis-plus.version>3.5.5</mybatis-plus.version>
|
||||
<poi.version>3.9</poi.version>
|
||||
<dozer.version>5.5.1</dozer.version>
|
||||
</properties>
|
||||
<dependencies>
|
||||
<dependency>
|
||||
|
@ -200,11 +202,23 @@
|
|||
<groupId>org.apache.hadoop</groupId>
|
||||
<artifactId>hadoop-client</artifactId>
|
||||
<version>3.3.6</version>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>org.slf4j</groupId>
|
||||
<artifactId>slf4j-reload4j</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.apache.hadoop</groupId>
|
||||
<artifactId>hadoop-hdfs</artifactId>
|
||||
<version>3.3.6</version>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>org.slf4j</groupId>
|
||||
<artifactId>slf4j-reload4j</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
|
||||
<!-- Swagger UI -->
|
||||
|
@ -264,6 +278,12 @@
|
|||
<groupId>org.elasticsearch</groupId>
|
||||
<artifactId>elasticsearch</artifactId>
|
||||
<version>7.14.0</version>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>org.apache.logging.log4j</groupId>
|
||||
<artifactId>log4j-api</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.elasticsearch.client</groupId>
|
||||
|
@ -383,8 +403,110 @@
|
|||
</dependency>
|
||||
|
||||
|
||||
<!-- Apache POI -->
|
||||
<dependency>
|
||||
<groupId>org.apache.poi</groupId>
|
||||
<artifactId>poi-ooxml</artifactId>
|
||||
<version>5.2.3</version>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>org.apache.logging.log4j</groupId>
|
||||
<artifactId>log4j-api</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.apache.poi</groupId>
|
||||
<artifactId>poi-scratchpad</artifactId>
|
||||
<version>5.2.3</version>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>org.apache.logging.log4j</groupId>
|
||||
<artifactId>log4j-api</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
<!-- 图片处理库 -->
|
||||
<dependency>
|
||||
<groupId>commons-io</groupId>
|
||||
<artifactId>commons-io</artifactId>
|
||||
<version>2.18.0</version>
|
||||
</dependency>
|
||||
|
||||
|
||||
<!-- poi office -->
|
||||
<!-- <dependency>
|
||||
<groupId>org.apache.poi</groupId>
|
||||
<artifactId>poi</artifactId>
|
||||
<version>${poi.version}</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.apache.poi</groupId>
|
||||
<artifactId>poi-ooxml</artifactId>
|
||||
<version>${poi.version}</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.apache.poi</groupId>
|
||||
<artifactId>poi-ooxml-schemas</artifactId>
|
||||
<version>${poi.version}</version>
|
||||
</dependency>-->
|
||||
|
||||
|
||||
<dependency>
|
||||
<groupId>net.sf.dozer</groupId>
|
||||
<artifactId>dozer</artifactId>
|
||||
<version>${dozer.version}</version>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>commons-collections</groupId>
|
||||
<artifactId>commons-collections</artifactId>
|
||||
</exclusion>
|
||||
<exclusion>
|
||||
<groupId>org.slf4j</groupId>
|
||||
<artifactId>slf4j-api</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
|
||||
|
||||
<dependency>
|
||||
<groupId>dev.langchain4j</groupId>
|
||||
<artifactId>langchain4j-community-dashscope-spring-boot-starter</artifactId>
|
||||
<version>1.0.0-beta2</version>
|
||||
<!-- <exclusions>
|
||||
<!– 排除新依赖中的 logback-classic –>
|
||||
<exclusion>
|
||||
<groupId>ch.qos.logback</groupId>
|
||||
<artifactId>logback-classic</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>-->
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.apache.tika</groupId>
|
||||
<artifactId>tika-core</artifactId>
|
||||
<version>3.1.0</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.apache.tika</groupId>
|
||||
<artifactId>tika-parsers-standard-package</artifactId>
|
||||
<version>3.1.0</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-webflux</artifactId>
|
||||
</dependency>
|
||||
|
||||
</dependencies>
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
<build>
|
||||
<plugins>
|
||||
<plugin>
|
||||
|
|
10
readme.md
10
readme.md
|
@ -1,3 +1,11 @@
|
|||
毕设后端
|
||||
<br>
|
||||
如果运行 暂时屏蔽face包
|
||||
如果运行 暂时屏蔽face包
|
||||
|
||||
|
||||
mvn dependency:tree > deps.txt
|
||||
<br>
|
||||
分析依赖
|
||||
这次主要是日志的
|
||||
slf4j-reload4
|
||||
log4j-api
|
|
@ -9,7 +9,7 @@ import org.springframework.cloud.openfeign.EnableFeignClients;
|
|||
|
||||
@Slf4j
|
||||
@SpringBootApplication
|
||||
@MapperScan("com.guwan.backend.mapper")
|
||||
@MapperScan({"com.guwan.backend.mapper", "com.guwan.backend.model.exam.mapper"})
|
||||
@EnableFeignClients
|
||||
public class BackendApplication {
|
||||
|
||||
|
@ -20,6 +20,7 @@ public class BackendApplication {
|
|||
|
||||
@PostConstruct
|
||||
public void init() {
|
||||
System.out.println("SLF4J Implementation: " + org.slf4j.LoggerFactory.getILoggerFactory().getClass().getName());
|
||||
log.info("大爱仙尊: http://localhost:8084/daxz.html?id=1");
|
||||
}
|
||||
|
||||
|
|
|
@ -0,0 +1,28 @@
|
|||
package com.guwan.backend;
|
||||
|
||||
import org.apache.tika.metadata.Metadata;
|
||||
import org.apache.tika.parser.AutoDetectParser;
|
||||
import org.apache.tika.parser.ParseContext;
|
||||
import org.apache.tika.parser.Parser;
|
||||
import org.apache.tika.sax.BodyContentHandler;
|
||||
|
||||
import java.io.FileInputStream;
|
||||
import java.io.InputStream;
|
||||
import java.net.URL;
|
||||
|
||||
public class VideoDuration {
|
||||
public static void main(String[] args) {
|
||||
String preSignedUrl = "http://localhost:9000/videos/ffffad37-9804-4765-ae18-3f8dcda9bea8.mp4"; // Minio 预签名 URL
|
||||
|
||||
try (InputStream stream = new URL(preSignedUrl).openStream()) {
|
||||
Metadata metadata = new Metadata();
|
||||
Parser parser = new AutoDetectParser();
|
||||
parser.parse(stream, new BodyContentHandler(), metadata, new ParseContext());
|
||||
|
||||
String duration = metadata.get("duration");
|
||||
System.out.println("视频时长(毫秒): " + duration);
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -76,6 +76,7 @@ public class OperationLogAspect {
|
|||
|
||||
// 获取当前用户信息
|
||||
Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
|
||||
//Java 16+ 模式匹配写法
|
||||
if (authentication != null && authentication.getPrincipal() instanceof CustomUserDetails userDetails) {
|
||||
sysLog.setUserId(userDetails.getUserId());
|
||||
sysLog.setUsername(userDetails.getUsername());
|
||||
|
|
|
@ -3,7 +3,7 @@ package com.guwan.backend.constant;
|
|||
import java.util.List;
|
||||
|
||||
/**
|
||||
* 安全相关常量配置
|
||||
* 本地缓存相关常量配置
|
||||
*/
|
||||
public class CacheConstants {
|
||||
|
||||
|
|
|
@ -0,0 +1,15 @@
|
|||
package com.guwan.backend.controller;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
@Data
|
||||
@AllArgsConstructor
|
||||
@NoArgsConstructor
|
||||
public class Ac{
|
||||
List<String> ids;
|
||||
String str;
|
||||
}
|
|
@ -11,15 +11,36 @@ import com.guwan.backend.mongodb.MongodbUserService;
|
|||
import com.guwan.backend.mongodb.User;
|
||||
import com.guwan.backend.pojo.entity.BookContent;
|
||||
import com.guwan.backend.util.MinioUtil;
|
||||
import dev.langchain4j.community.model.dashscope.QwenChatModel;
|
||||
import dev.langchain4j.community.model.dashscope.QwenStreamingChatModel;
|
||||
import dev.langchain4j.model.chat.response.ChatResponse;
|
||||
import dev.langchain4j.model.chat.response.StreamingChatResponseHandler;
|
||||
import io.minio.MinioClient;
|
||||
import lombok.RequiredArgsConstructor;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import okhttp3.OkHttpClient;
|
||||
import okhttp3.Request;
|
||||
import okhttp3.Response;
|
||||
import org.apache.poi.xslf.usermodel.XMLSlideShow;
|
||||
import org.apache.poi.xslf.usermodel.XSLFSlide;
|
||||
import org.apache.tika.metadata.Metadata;
|
||||
import org.apache.tika.parser.AutoDetectParser;
|
||||
import org.apache.tika.parser.ParseContext;
|
||||
import org.apache.tika.parser.Parser;
|
||||
import org.apache.tika.sax.BodyContentHandler;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
import org.springframework.web.multipart.MultipartFile;
|
||||
import reactor.core.publisher.Flux;
|
||||
|
||||
import javax.imageio.ImageIO;
|
||||
import java.awt.*;
|
||||
import java.awt.geom.Rectangle2D;
|
||||
import java.awt.image.BufferedImage;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.net.URL;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.regex.Matcher;
|
||||
|
@ -36,6 +57,8 @@ public class CommonController {
|
|||
private final VoiceServiceClient voiceServiceClient;
|
||||
private final SimpleTTSClient simpleTTSClient;
|
||||
|
||||
private final MinioClient minioClient;
|
||||
|
||||
|
||||
|
||||
|
||||
|
@ -44,6 +67,8 @@ public class CommonController {
|
|||
private final MongodbUserService mongodbUserService;
|
||||
|
||||
private final EveryReadDetailOfMongodbService everyReadDetailOfMongodbService;
|
||||
private final QwenChatModel qwenChatModel;
|
||||
private final QwenStreamingChatModel qwenStreamingChatModel;
|
||||
|
||||
@PostMapping("/uploadFile")
|
||||
public Result<String> uploadFile(String bucketName, MultipartFile file){
|
||||
|
@ -361,4 +386,135 @@ public class CommonController {
|
|||
}
|
||||
|
||||
|
||||
@GetMapping("/testQwen")
|
||||
public Result testQwen(
|
||||
@RequestParam(value = "str1", required = false) String message) {
|
||||
|
||||
var response = qwenChatModel.chat(message);
|
||||
System.out.println("response = " + response);
|
||||
return Result.success(response);
|
||||
}
|
||||
|
||||
|
||||
@GetMapping(value = "/testQwenStreaming", produces = "text/steam;charset=UTF-8")
|
||||
public Flux<String> testQwenStreaming(
|
||||
@RequestParam(value = "message", required = false) String message) {
|
||||
|
||||
Flux<String> flux = Flux.create(fluxSink -> {
|
||||
qwenStreamingChatModel.chat(message, new StreamingChatResponseHandler() {
|
||||
//每一次流式响应的文本
|
||||
@Override
|
||||
public void onPartialResponse(String partialResponse) {
|
||||
fluxSink.next(partialResponse);
|
||||
}
|
||||
|
||||
//响应结束的文本
|
||||
@Override
|
||||
public void onCompleteResponse(ChatResponse chatResponse) {
|
||||
fluxSink.complete();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onError(Throwable throwable) {
|
||||
fluxSink.error(throwable);
|
||||
}
|
||||
});
|
||||
|
||||
});
|
||||
return flux;
|
||||
|
||||
}
|
||||
|
||||
|
||||
@GetMapping("/PPTToImageConverter")
|
||||
public void PPTToImageConverter() throws IOException {
|
||||
|
||||
String pptFile = "D:\\00_桌面\\公司简介.pptx"; // PPT 文件路径
|
||||
String outputDir = "output_images"; // 输出目录
|
||||
|
||||
FileInputStream inputStream = new FileInputStream(pptFile);
|
||||
XMLSlideShow ppt = new XMLSlideShow(inputStream);
|
||||
inputStream.close();
|
||||
|
||||
// 创建输出目录
|
||||
File dir = new File(outputDir);
|
||||
if (!dir.exists()) {
|
||||
dir.mkdirs();
|
||||
}
|
||||
|
||||
// 获取幻灯片
|
||||
int slideNumber = 1;
|
||||
for (XSLFSlide slide : ppt.getSlides()) {
|
||||
Dimension pgsize = ppt.getPageSize();
|
||||
BufferedImage img = new BufferedImage(pgsize.width, pgsize.height, BufferedImage.TYPE_INT_ARGB);
|
||||
Graphics2D graphics = img.createGraphics();
|
||||
|
||||
// 设置背景为白色
|
||||
graphics.setPaint(Color.WHITE);
|
||||
graphics.fill(new Rectangle2D.Float(0, 0, pgsize.width, pgsize.height));
|
||||
|
||||
// 渲染幻灯片
|
||||
slide.draw(graphics);
|
||||
|
||||
// 输出为图片
|
||||
File outputFile = new File(outputDir + "/slide_" + slideNumber + ".png");
|
||||
ImageIO.write(img, "png", outputFile);
|
||||
System.out.println("Saved slide: " + outputFile.getAbsolutePath());
|
||||
|
||||
slideNumber++;
|
||||
}
|
||||
ppt.close();
|
||||
}
|
||||
|
||||
|
||||
@GetMapping("/videoDuration")
|
||||
public void VideoDuration() throws IOException {
|
||||
|
||||
String preSignedUrl = "http://localhost:9000/videos/ffffad37-9804-4765-ae18-3f8dcda9bea8.mp4"; // Minio 预签名 URL
|
||||
|
||||
|
||||
// minioClient.getObject()
|
||||
InputStream stream = minioUtil.getFileInputStream("videos", "ffffad37-9804-4765-ae18-3f8dcda9bea8.mp4");
|
||||
|
||||
|
||||
System.out.println("stream = " + stream);
|
||||
try {
|
||||
Metadata metadata = new Metadata();
|
||||
Parser parser = new AutoDetectParser();
|
||||
parser.parse(stream, new BodyContentHandler(), metadata, new ParseContext());
|
||||
|
||||
String duration = metadata.get("duration");
|
||||
System.out.println("视频时长(毫秒): " + duration);
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@PostMapping("/testPostUseParam")
|
||||
//测试公司写法
|
||||
public void testPostUseParam(@RequestParam("ids") List<String> ids,
|
||||
@RequestParam("str") String str){
|
||||
System.out.println("ids = " + ids);
|
||||
System.out.println("str = " + str);
|
||||
}
|
||||
|
||||
@PostMapping("/testPostUseBody")
|
||||
public void testPostUseBody(@RequestBody Ac ac){
|
||||
System.out.println("ac = " + ac);
|
||||
}
|
||||
|
||||
@GetMapping("/testGetParam")
|
||||
public void testGetParam(@RequestParam("ids") List<String> ids,
|
||||
@RequestParam("str") String str){
|
||||
System.out.println("ids = " + ids);
|
||||
System.out.println("str = " + str);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -0,0 +1,27 @@
|
|||
package com.guwan.backend.controller;
|
||||
|
||||
class CompileResult {
|
||||
boolean success;
|
||||
String message;
|
||||
|
||||
// constructor...
|
||||
}
|
||||
|
||||
class CaseResult {
|
||||
boolean passed;
|
||||
String input;
|
||||
String expected;
|
||||
String actual;
|
||||
String message;
|
||||
|
||||
// 成功构造器
|
||||
public CaseResult(boolean passed, String input,
|
||||
String expected, String actual) {
|
||||
// ...
|
||||
}
|
||||
|
||||
// 失败构造器
|
||||
public CaseResult(boolean passed, String message) {
|
||||
// ...
|
||||
}
|
||||
}
|
|
@ -2,6 +2,8 @@ package com.guwan.backend.controller;
|
|||
|
||||
|
||||
|
||||
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||
import com.guwan.backend.common.Result;
|
||||
import com.guwan.backend.mongodb.CategoryService;
|
||||
import com.guwan.backend.pojo.Courses;
|
||||
|
@ -9,7 +11,7 @@ import com.guwan.backend.pojo.dto.BSCategory;
|
|||
import com.guwan.backend.service.BSCategoryService;
|
||||
import com.guwan.backend.service.CoursesService;
|
||||
import lombok.RequiredArgsConstructor;
|
||||
import org.springframework.data.domain.Page;
|
||||
|
||||
import org.springframework.data.domain.PageRequest;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
|
@ -45,11 +47,18 @@ public class CoursesController {
|
|||
* @param pageRequest 分页对象
|
||||
* @return 查询结果
|
||||
*/
|
||||
@GetMapping
|
||||
public Result queryByPage() {
|
||||
@GetMapping("/homePage")
|
||||
public Result queryByPage(@RequestParam("page") Integer pageNumber,
|
||||
@RequestParam("size") Integer size) {
|
||||
// return Result.success(this.coursesService.queryByPage(courses, pageRequest));
|
||||
|
||||
return Result.success(this.coursesService.list().stream()
|
||||
Page<Courses> page = new Page<>(pageNumber, size);
|
||||
|
||||
LambdaQueryWrapper<Courses> lambdaQueryWrapper = new LambdaQueryWrapper<>();
|
||||
|
||||
Page<Courses> resultPage = this.coursesService.page(page, lambdaQueryWrapper);
|
||||
|
||||
return Result.success(resultPage.getRecords().stream()
|
||||
.peek(course -> {
|
||||
course.setCategoryName(categoryService.list()
|
||||
.stream()
|
||||
|
|
|
@ -12,7 +12,6 @@ import com.mongodb.client.gridfs.model.GridFSFile;
|
|||
import io.minio.MinioClient;
|
||||
import lombok.RequiredArgsConstructor;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.apache.hadoop.thirdparty.protobuf.Api;
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.validation.annotation.Validated;
|
||||
|
@ -21,8 +20,6 @@ import org.springframework.web.client.RestTemplate;
|
|||
import org.springframework.web.multipart.MultipartFile;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
@Slf4j
|
||||
|
|
|
@ -1,19 +1,26 @@
|
|||
package com.guwan.backend.controller;
|
||||
|
||||
import com.github.dockerjava.api.command.CreateContainerCmd;
|
||||
import com.github.dockerjava.api.command.CreateContainerResponse;
|
||||
import com.github.dockerjava.api.command.ExecCreateCmdResponse;
|
||||
import com.github.dockerjava.api.model.*;
|
||||
import com.github.dockerjava.core.DockerClientBuilder;
|
||||
import com.github.dockerjava.core.command.ExecStartResultCallback;
|
||||
import com.github.dockerjava.httpclient5.ApacheDockerHttpClient;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
import com.github.dockerjava.api.DockerClient;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.file.Path;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
import java.util.*;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
@RestController
|
||||
@RequestMapping("/api/common")
|
||||
|
@ -96,6 +103,217 @@ public class DockerController {
|
|||
|
||||
}
|
||||
|
||||
private static final String IMAGE_NAME = "gcc:latest";
|
||||
private static final int TIMEOUT_SECONDS = 10;
|
||||
|
||||
/* @GetMapping("/testCpp")
|
||||
public void testCpp() throws IOException {
|
||||
|
||||
System.out.println("dockerClient = " + dockerClient);
|
||||
|
||||
|
||||
try {
|
||||
// 创建容器
|
||||
CreateContainerResponse container = dockerClient.createContainerCmd(IMAGE_NAME)
|
||||
.withTty(true)
|
||||
.withWorkingDir("/app")
|
||||
.exec();
|
||||
|
||||
String containerId = container.getId();
|
||||
dockerClient.startContainerCmd(containerId).exec();
|
||||
|
||||
// 写入代码(使用cat和here-doc避免转义问题)
|
||||
String cppCode = "#include <iostream>\n"
|
||||
+ "int main() { std::cout << \"Hello Docker!\\n\"; return 0; }";
|
||||
|
||||
execCommand(dockerClient, containerId,
|
||||
"sh -c 'cat > /app/main.cpp <<EOF\n" + cppCode + "\nEOF'");
|
||||
|
||||
// 编译代码(使用相对路径)
|
||||
execCommand(dockerClient, containerId, "g++ main.cpp -o main");
|
||||
|
||||
// 执行程序(使用相对路径)
|
||||
String output = execCommand(dockerClient, containerId, "./main");
|
||||
System.out.println("程序输出: " + output);
|
||||
|
||||
} catch (Exception e) {
|
||||
throw new RuntimeException(e);
|
||||
} finally {
|
||||
// 清理容器
|
||||
*//*dockerClient.listContainersCmd().withShowAll(true).exec().forEach(c ->
|
||||
dockerClient.removeContainerCmd(c.getId()).withForce(true).exec());*//*
|
||||
//dockerClient.close();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private static String execCommand(DockerClient dockerClient, String containerId, String command)
|
||||
throws Exception {
|
||||
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
|
||||
ExecCreateCmdResponse exec = dockerClient.execCreateCmd(containerId)
|
||||
.withCmd("sh", "-c", command)
|
||||
.withAttachStdout(true)
|
||||
.withAttachStderr(true)
|
||||
.exec();
|
||||
|
||||
dockerClient.execStartCmd(exec.getId())
|
||||
.exec(new ExecStartResultCallback(outputStream, System.err))
|
||||
.awaitCompletion(TIMEOUT_SECONDS, TimeUnit.SECONDS);
|
||||
|
||||
return outputStream.toString().trim();
|
||||
}*/
|
||||
|
||||
|
||||
@GetMapping("/judge")
|
||||
public ResponseEntity<Map<String, Object>> judgeCode() {
|
||||
Map<String, Object> result = new HashMap<>();
|
||||
List<TestCase> testCases = new ArrayList<>();
|
||||
|
||||
// 真实用户提交的C++代码(示例:A+B问题)
|
||||
String cppCode =
|
||||
"#include <iostream>\n" +
|
||||
"using namespace std;\n" +
|
||||
"int main() {\n" +
|
||||
" int a, b;\n" +
|
||||
" cin >> a >> b;\n" +
|
||||
" cout << a + b << endl;\n" +
|
||||
" return 0;\n" +
|
||||
"}";
|
||||
|
||||
// 定义测试用例
|
||||
testCases.add(new TestCase("1 2", "3")); // 基础测试
|
||||
testCases.add(new TestCase("-5 8", "3")); // 负数测试
|
||||
testCases.add(new TestCase("1000000 2000000", "3000000")); // 大数测试
|
||||
testCases.add(new TestCase("", "")); // 错误输入测试
|
||||
testCases.add(new TestCase("3", "")); // 不完整输入测试
|
||||
|
||||
String containerId = null;
|
||||
try {
|
||||
// 创建容器(使用官方GCC镜像)
|
||||
CreateContainerResponse container = dockerClient.createContainerCmd("gcc:latest")
|
||||
.withTty(true)
|
||||
.withWorkingDir("/app")
|
||||
.withHostConfig(HostConfig.newHostConfig()
|
||||
.withMemory(100 * 1024 * 1024L) // 限制内存100MB
|
||||
.withCpuCount(1L)) // 限制1核CPU
|
||||
.exec();
|
||||
|
||||
containerId = container.getId();
|
||||
dockerClient.startContainerCmd(containerId).exec();
|
||||
|
||||
// 1. 写入代码文件
|
||||
execCommand(containerId, "mkdir -p /app", 10);
|
||||
execCommand(containerId,
|
||||
"bash -c 'cat > /app/main.cpp <<EOF\n" +
|
||||
cppCode.replace("'", "'\"'\"'") + // 处理单引号
|
||||
"\nEOF'", 10);
|
||||
|
||||
// 2. 编译代码(带编译错误检查)
|
||||
String compileOutput = execCommand(containerId, "g++ main.cpp -o main -O2 -Wall", 10);
|
||||
if (!compileOutput.isEmpty()) {
|
||||
result.put("status", "compile_error");
|
||||
result.put("message", compileOutput);
|
||||
return ResponseEntity.ok(result);
|
||||
}
|
||||
|
||||
// 3. 执行测试用例
|
||||
List<Map<String, Object>> caseResults = new ArrayList<>();
|
||||
for (int i = 0; i < testCases.size(); i++) {
|
||||
TestCase testCase = testCases.get(i);
|
||||
Map<String, Object> caseResult = new HashMap<>();
|
||||
caseResult.put("case_id", i + 1);
|
||||
caseResult.put("input", testCase.input);
|
||||
caseResult.put("expected", testCase.expectedOutput);
|
||||
|
||||
try {
|
||||
// 写入输入文件
|
||||
execCommand(containerId,
|
||||
"bash -c 'cat > /app/input.txt <<EOF\n" +
|
||||
testCase.input.replace("'", "'\"'\"'") +
|
||||
"\nEOF'", 10);
|
||||
|
||||
// 执行程序(带超时控制)
|
||||
String actualOutput = execCommand(containerId,
|
||||
"timeout 2s ./main < /app/input.txt 2>&1", // 捕获标准错误
|
||||
3 // 超时3秒
|
||||
);
|
||||
|
||||
// 标准化输出处理
|
||||
actualOutput = actualOutput.trim()
|
||||
.replaceAll("\r\n", "\n")
|
||||
.replaceAll("[ \\t]+", " ");
|
||||
|
||||
// 结果对比
|
||||
boolean isPassed = actualOutput.equals(testCase.expectedOutput);
|
||||
caseResult.put("status", isPassed ? "passed" : "failed");
|
||||
caseResult.put("actual", actualOutput);
|
||||
|
||||
// 特殊错误类型检测
|
||||
if (actualOutput.contains("Timeout")) {
|
||||
caseResult.put("status", "time_limit_exceeded");
|
||||
} else if (actualOutput.contains("runtime error")) {
|
||||
caseResult.put("status", "runtime_error");
|
||||
} else if (actualOutput.isEmpty()) {
|
||||
caseResult.put("status", "no_output");
|
||||
}
|
||||
|
||||
} catch (Exception e) {
|
||||
caseResult.put("status", "system_error");
|
||||
caseResult.put("message", e.getMessage());
|
||||
}
|
||||
|
||||
caseResults.add(caseResult);
|
||||
}
|
||||
|
||||
result.put("status", "completed");
|
||||
result.put("cases", caseResults);
|
||||
result.put("pass_count", caseResults.stream()
|
||||
.filter(c -> "passed".equals(c.get("status")))
|
||||
.count());
|
||||
|
||||
} catch (Exception e) {
|
||||
result.put("status", "system_error");
|
||||
result.put("message", e.getMessage());
|
||||
} finally {
|
||||
if (containerId != null) {
|
||||
dockerClient.removeContainerCmd(containerId)
|
||||
.withForce(true)
|
||||
.exec();
|
||||
}
|
||||
}
|
||||
|
||||
return ResponseEntity.ok(result);
|
||||
}
|
||||
|
||||
// 辅助方法:执行容器命令
|
||||
private String execCommand(String containerId, String command, int timeoutSeconds)
|
||||
throws Exception {
|
||||
|
||||
ByteArrayOutputStream output = new ByteArrayOutputStream();
|
||||
ExecCreateCmdResponse exec = dockerClient.execCreateCmd(containerId)
|
||||
.withCmd("bash", "-c", command)
|
||||
.withAttachStdout(true)
|
||||
.withAttachStderr(true)
|
||||
.exec();
|
||||
|
||||
dockerClient.execStartCmd(exec.getId())
|
||||
.exec(new ExecStartResultCallback(output, output))
|
||||
.awaitCompletion(timeoutSeconds, TimeUnit.SECONDS);
|
||||
|
||||
return output.toString().trim();
|
||||
}
|
||||
|
||||
// 测试用例类
|
||||
static class TestCase {
|
||||
String input;
|
||||
String expectedOutput;
|
||||
|
||||
public TestCase(String input, String expectedOutput) {
|
||||
this.input = input;
|
||||
this.expectedOutput = expectedOutput;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
|
|
@ -0,0 +1,79 @@
|
|||
/*
|
||||
package com.guwan.backend.controller;
|
||||
|
||||
import com.github.dockerjava.api.DockerClient;
|
||||
import com.github.dockerjava.api.command.CreateContainerResponse;
|
||||
import com.github.dockerjava.api.command.ExecCreateCmdResponse;
|
||||
import com.github.dockerjava.core.DockerClientBuilder;
|
||||
import com.github.dockerjava.core.command.ExecStartResultCallback;
|
||||
import com.github.dockerjava.httpclient5.ApacheDockerHttpClient;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
public class DockerCppRunner {
|
||||
|
||||
private static final String IMAGE_NAME = "cpp-runner";
|
||||
private static final int TIMEOUT_SECONDS = 10;
|
||||
|
||||
private final DockerClient dockerClient;
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
DockerClient dockerClient = DockerClientBuilder.getInstance()
|
||||
.withDockerHttpClient(
|
||||
new ApacheDockerHttpClient.Builder()
|
||||
.dockerHost(DockerClientBuilder.getDefaultDockerHost())
|
||||
.build()
|
||||
).build();
|
||||
|
||||
try {
|
||||
// 创建容器
|
||||
CreateContainerResponse container = dockerClient.createContainerCmd(IMAGE_NAME)
|
||||
.withTty(true)
|
||||
.exec();
|
||||
|
||||
String containerId = container.getId();
|
||||
|
||||
// 启动容器
|
||||
dockerClient.startContainerCmd(containerId).exec();
|
||||
|
||||
// 写入 C++ 代码到容器内
|
||||
String cppCode = "#include <iostream>\n"
|
||||
+ "int main() { std::cout << \"Hello Docker!\\n\"; return 0; }";
|
||||
|
||||
// 将代码写入容器中的文件
|
||||
dockerClient.execCreateCmd(containerId)
|
||||
.withCmd("sh", "-c", "echo '" + cppCode + "' > /app/main.cpp")
|
||||
.exec();
|
||||
|
||||
// 编译 C++ 代码
|
||||
execCommand(dockerClient, containerId, "g++ /app/main.cpp -o /app/main");
|
||||
|
||||
// 运行编译后的程序
|
||||
String output = execCommand(dockerClient, containerId, "/app/main");
|
||||
System.out.println("程序输出: " + output);
|
||||
|
||||
} finally {
|
||||
// 清理容器
|
||||
dockerClient.listContainersCmd().withShowAll(true).exec().forEach(c ->
|
||||
dockerClient.removeContainerCmd(c.getId()).withForce(true).exec());
|
||||
dockerClient.close();
|
||||
}
|
||||
}
|
||||
|
||||
private static String execCommand(DockerClient dockerClient, String containerId, String command)
|
||||
throws Exception {
|
||||
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
|
||||
ExecCreateCmdResponse exec = dockerClient.execCreateCmd(containerId)
|
||||
.withCmd("sh", "-c", command)
|
||||
.withAttachStdout(true)
|
||||
.withAttachStderr(true)
|
||||
.exec();
|
||||
|
||||
dockerClient.execStartCmd(exec.getId())
|
||||
.exec(new ExecStartResultCallback(outputStream, System.err))
|
||||
.awaitCompletion(TIMEOUT_SECONDS, TimeUnit.SECONDS);
|
||||
|
||||
return outputStream.toString().trim();
|
||||
}
|
||||
}*/
|
|
@ -0,0 +1,13 @@
|
|||
package com.guwan.backend.controller;
|
||||
|
||||
// 测试用例类
|
||||
class TestCase {
|
||||
String input; // 输入数据
|
||||
String expectedOutput; // 预期输出
|
||||
int timeoutSeconds = 2;// 超时时间
|
||||
|
||||
public TestCase(String input, String expectedOutput) {
|
||||
this.input = input;
|
||||
this.expectedOutput = expectedOutput;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,59 @@
|
|||
package com.guwan.backend.core;
|
||||
|
||||
import org.dozer.DozerBeanMapper;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
import java.util.List;
|
||||
import java.util.function.Function;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
|
||||
/**
|
||||
* 简单封装Dozer, 实现深度转换Bean<->Bean的Mapper.实现:
|
||||
*
|
||||
* 1. 持有Mapper的单例.
|
||||
* 2. 返回值类型转换.
|
||||
* 3. 批量转换Collection中的所有对象.
|
||||
* 4. 区分创建新的B对象与将对象A值复制到已存在的B对象两种函数.
|
||||
*
|
||||
*/
|
||||
public class BeanMapper {
|
||||
|
||||
/**
|
||||
* 持有Dozer单例, 避免重复创建DozerMapper消耗资源.
|
||||
*/
|
||||
private static DozerBeanMapper dozerBeanMapper = new DozerBeanMapper();
|
||||
|
||||
/**
|
||||
* 基于Dozer转换对象的类型.
|
||||
*/
|
||||
public static <T> T map(Object source, Class<T> destinationClass) {
|
||||
return dozerBeanMapper.map(source, destinationClass);
|
||||
}
|
||||
|
||||
/**
|
||||
* 基于Dozer转换Collection中对象的类型.
|
||||
*/
|
||||
public static <T> List<T> mapList(Iterable<?> sourceList, Class<T> destinationClass) {
|
||||
List<T> destinationList = new ArrayList();
|
||||
for (Object sourceObject : sourceList) {
|
||||
T destinationObject = dozerBeanMapper.map(sourceObject, destinationClass);
|
||||
destinationList.add(destinationObject);
|
||||
}
|
||||
return destinationList;
|
||||
}
|
||||
|
||||
/**
|
||||
* 基于Dozer将对象A的值拷贝到对象B中.
|
||||
*/
|
||||
public static void copy(Object source, Object destinationObject) {
|
||||
if(source!=null) {
|
||||
dozerBeanMapper.map(source, destinationObject);
|
||||
}
|
||||
}
|
||||
|
||||
public static <T, S> List<T> mapList(Collection<S> source, Function<? super S, ? extends T> mapper) {
|
||||
return source.stream().map(mapper).collect(Collectors.toList());
|
||||
}
|
||||
}
|
|
@ -0,0 +1,21 @@
|
|||
package com.guwan.backend.core.annon;
|
||||
|
||||
import java.lang.annotation.ElementType;
|
||||
import java.lang.annotation.Retention;
|
||||
import java.lang.annotation.RetentionPolicy;
|
||||
import java.lang.annotation.Target;
|
||||
|
||||
/**
|
||||
* 数据字典注解
|
||||
* @author bool
|
||||
*/
|
||||
@Target(ElementType.FIELD)
|
||||
@Retention(RetentionPolicy.RUNTIME)
|
||||
public @interface Dict {
|
||||
|
||||
String dicCode();
|
||||
|
||||
String dicText() default "";
|
||||
|
||||
String dictTable() default "";
|
||||
}
|
|
@ -0,0 +1,67 @@
|
|||
package com.guwan.backend.core.api;
|
||||
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* 全局错误码定义,用于定义接口的响应数据,
|
||||
* 枚举名称全部使用代码命名,在系统中调用,免去取名难的问题。
|
||||
* @author bool
|
||||
* @date 2019-06-14 21:15
|
||||
*/
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public enum ApiError implements Serializable {
|
||||
|
||||
|
||||
/**
|
||||
* 通用错误,接口参数不全
|
||||
*/
|
||||
ERROR_10010001("参数不全或类型错误!"),
|
||||
ERROR_10010002("您还未登录,请先登录!"),
|
||||
ERROR_10010003("数据不存在!"),
|
||||
ERROR_10010012("图形验证码错误!"),
|
||||
ERROR_10010013("短信验证码错误!"),
|
||||
ERROR_10010014("不允许重复评论!"),
|
||||
|
||||
/**
|
||||
* 考试相关错误
|
||||
*/
|
||||
ERROR_20010001("试题被删除,无法继续考试!"),
|
||||
ERROR_20010002("您有正在进行的考试!"),
|
||||
|
||||
|
||||
ERROR_90010001("账号不存在,请确认!"),
|
||||
ERROR_90010002("账号或密码错误!"),
|
||||
ERROR_90010003("至少要包含一个角色!"),
|
||||
ERROR_90010004("管理员账号无法修改!"),
|
||||
ERROR_90010005("账号被禁用,请联系管理员!"),
|
||||
ERROR_90010006("活动用户不足,无法开启竞拍!"),
|
||||
ERROR_90010007("旧密码不正确,请确认!"),
|
||||
|
||||
|
||||
ERROR_60000001("数据不存在!");
|
||||
|
||||
public String msg;
|
||||
|
||||
/**
|
||||
* 生成Markdown格式文档,用于更新文档用的
|
||||
* @param args
|
||||
*/
|
||||
public static void main(String[] args) {
|
||||
for (ApiError e : ApiError.values()) {
|
||||
System.out.println("'"+e.name().replace("ERROR_", "")+"':'"+e.msg+"',");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取错误码
|
||||
* @return
|
||||
*/
|
||||
public Integer getCode(){
|
||||
return Integer.parseInt(this.name().replace("ERROR_", ""));
|
||||
}
|
||||
}
|
|
@ -0,0 +1,64 @@
|
|||
package com.guwan.backend.core.api;
|
||||
|
||||
|
||||
|
||||
import com.guwan.backend.core.exception.ServiceException;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* 数据结果返回的封装
|
||||
* @author bool
|
||||
* @date 2018/11/20 09:48
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@ApiModel(value="接口响应", description="接口响应")
|
||||
public class ApiRest<T>{
|
||||
|
||||
/**
|
||||
* 响应消息
|
||||
*/
|
||||
@ApiModelProperty(value = "响应消息")
|
||||
private String msg;
|
||||
/**
|
||||
* 响应代码
|
||||
*/
|
||||
@ApiModelProperty(value = "响应代码,0为成功,1为失败", required = true)
|
||||
private Integer code;
|
||||
|
||||
/**
|
||||
* 请求或响应body
|
||||
*/
|
||||
@ApiModelProperty(value = "响应内容")
|
||||
protected T data;
|
||||
|
||||
|
||||
/**
|
||||
* 是否成功
|
||||
* @return
|
||||
*/
|
||||
public boolean isSuccess(){
|
||||
return code.equals(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
* @param error
|
||||
*/
|
||||
public ApiRest(ServiceException error){
|
||||
this.code = error.getCode();
|
||||
this.msg = error.getMsg();
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
* @param error
|
||||
*/
|
||||
public ApiRest(ApiError error){
|
||||
this.code = error.getCode();
|
||||
this.msg = error.msg;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,154 @@
|
|||
package com.guwan.backend.core.api.controller;
|
||||
|
||||
|
||||
import com.guwan.backend.core.api.ApiError;
|
||||
import com.guwan.backend.core.api.ApiRest;
|
||||
import com.guwan.backend.core.exception.ServiceException;
|
||||
|
||||
/**
|
||||
* 基础控制器
|
||||
* @author Dav
|
||||
*/
|
||||
public class BaseController {
|
||||
|
||||
/**
|
||||
* 成功默认消息
|
||||
*/
|
||||
private static final Integer CODE_SUCCESS = 0;
|
||||
private static final String MSG_SUCCESS = "操作成功!";
|
||||
|
||||
/**
|
||||
* 失败默认消息
|
||||
*/
|
||||
private static final Integer CODE_FAILURE = 1;
|
||||
private static final String MSG_FAILURE = "请求失败!";
|
||||
|
||||
|
||||
/**
|
||||
* 完成消息构造
|
||||
* @param code
|
||||
* @param message
|
||||
* @param data
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> message(Integer code, String message, T data){
|
||||
ApiRest<T> response = new ApiRest<>();
|
||||
response.setCode(code);
|
||||
response.setMsg(message);
|
||||
if(data!=null) {
|
||||
response.setData(data);
|
||||
}
|
||||
return response;
|
||||
}
|
||||
|
||||
/**
|
||||
* 请求成功空数据
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> success(){
|
||||
return message(0, "请求成功!", null);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 请求成功,通用代码
|
||||
* @param message
|
||||
* @param data
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> success(String message, T data){
|
||||
return message(CODE_SUCCESS, message, data);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 请求成功,仅内容
|
||||
* @param data
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> success(T data){
|
||||
return message(CODE_SUCCESS, MSG_SUCCESS, data);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 请求失败,完整构造
|
||||
* @param code
|
||||
* @param message
|
||||
* @param data
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(Integer code, String message, T data){
|
||||
return message(code, message, data);
|
||||
}
|
||||
|
||||
/**
|
||||
* 请求失败,消息和内容
|
||||
* @param message
|
||||
* @param data
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(String message, T data){
|
||||
return message(CODE_FAILURE, message, data);
|
||||
}
|
||||
|
||||
/**
|
||||
* 请求失败,消息
|
||||
* @param message
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(String message){
|
||||
return message(CODE_FAILURE, message, null);
|
||||
}
|
||||
|
||||
/**
|
||||
* 请求失败,仅内容
|
||||
* @param data
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(T data){
|
||||
return message(CODE_FAILURE, MSG_FAILURE, data);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 请求失败,仅内容
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(){
|
||||
return message(CODE_FAILURE, MSG_FAILURE, null);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 请求失败,仅内容
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(ApiError error, T data){
|
||||
return message(error.getCode(), error.msg, data);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 请求失败,仅内容
|
||||
* @param ex
|
||||
* @param <T>
|
||||
* @return
|
||||
*/
|
||||
protected <T> ApiRest<T> failure(ServiceException ex){
|
||||
ApiRest<T> apiRest = message(ex.getCode(), ex.getMsg(), null);
|
||||
return apiRest;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,15 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* 请求和响应的基础类,用于处理序列化
|
||||
* @author dav
|
||||
* @date 2019/3/16 15:56
|
||||
*/
|
||||
@Data
|
||||
public class BaseDTO implements Serializable {
|
||||
|
||||
}
|
|
@ -0,0 +1,27 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 主键通用请求类,用于根据ID查询
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2019-04-20 12:15
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="主键通用请求类", description="主键通用请求类")
|
||||
public class BaseIdReqDTO extends BaseDTO {
|
||||
|
||||
|
||||
@ApiModelProperty(value = "主键ID", required=true)
|
||||
private String id;
|
||||
|
||||
@JsonIgnore
|
||||
private String userId;
|
||||
|
||||
}
|
|
@ -0,0 +1,25 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 主键通用响应类,用于添加后返回内容
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2019-04-20 12:15
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="主键通用响应类", description="主键通用响应类")
|
||||
@AllArgsConstructor
|
||||
@NoArgsConstructor
|
||||
public class BaseIdRespDTO extends BaseDTO {
|
||||
|
||||
@ApiModelProperty(value = "主键ID", required=true)
|
||||
private String id;
|
||||
}
|
|
@ -0,0 +1,25 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* 通用ID列表类操作,用于批量删除、修改状态等
|
||||
* @author bool
|
||||
* @date 2019-08-01 19:07
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="删除参数", description="删除参数")
|
||||
public class BaseIdsReqDTO extends BaseDTO {
|
||||
|
||||
|
||||
@JsonIgnore
|
||||
private String userId;
|
||||
|
||||
@ApiModelProperty(value = "要删除的ID列表", required = true)
|
||||
private List<String> ids;
|
||||
}
|
|
@ -0,0 +1,31 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 通用状态请求类,用于修改状态什么的
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2019-04-20 12:15
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="通用状态请求类", description="通用状态请求类")
|
||||
@AllArgsConstructor
|
||||
@NoArgsConstructor
|
||||
public class BaseStateReqDTO extends BaseDTO {
|
||||
|
||||
|
||||
@ApiModelProperty(value = "要修改对象的ID列表", required=true)
|
||||
private List<String> ids;
|
||||
|
||||
@ApiModelProperty(value = "通用状态,0为正常,1为禁用", required=true)
|
||||
private Integer state;
|
||||
}
|
|
@ -0,0 +1,47 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* 分页查询类
|
||||
* @param <T>
|
||||
* @author bool
|
||||
*/
|
||||
@ApiModel(value="分页参数", description="分页参数")
|
||||
@Data
|
||||
public class PagingReqDTO<T> {
|
||||
|
||||
|
||||
@ApiModelProperty(value = "当前页码", required = true, example = "1")
|
||||
private Integer current;
|
||||
|
||||
@ApiModelProperty(value = "每页数量", required = true, example = "10")
|
||||
private Integer size;
|
||||
|
||||
@ApiModelProperty(value = "查询参数")
|
||||
private T params;
|
||||
|
||||
@ApiModelProperty(value = "排序字符")
|
||||
private String orderBy;
|
||||
|
||||
@JsonIgnore
|
||||
@ApiModelProperty(value = "当前用户的ID")
|
||||
private String userId;
|
||||
|
||||
/**
|
||||
* 转换成MyBatis的简单分页对象
|
||||
* @return
|
||||
*/
|
||||
public Page toPage(){
|
||||
Page page = new Page();
|
||||
page.setCurrent(this.current);
|
||||
page.setSize(this.size);
|
||||
return page;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,30 @@
|
|||
package com.guwan.backend.core.api.dto;
|
||||
|
||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||
|
||||
/**
|
||||
* 分页响应类
|
||||
* @author bool
|
||||
* @date 2019-07-20 15:17
|
||||
* @param <T>
|
||||
*/
|
||||
public class PagingRespDTO<T> extends Page<T> {
|
||||
|
||||
/**
|
||||
* 获取页面总数量
|
||||
* @return
|
||||
*/
|
||||
@Override
|
||||
public long getPages() {
|
||||
if (this.getSize() == 0L) {
|
||||
return 0L;
|
||||
} else {
|
||||
long pages = this.getTotal() / this.getSize();
|
||||
if (this.getTotal() % this.getSize() != 0L) {
|
||||
++pages;
|
||||
}
|
||||
return pages;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,48 @@
|
|||
package com.guwan.backend.core.api.utils;
|
||||
|
||||
import com.alibaba.fastjson.serializer.SerializerFeature;
|
||||
import com.alibaba.fastjson.support.config.FastJsonConfig;
|
||||
import com.alibaba.fastjson.support.spring.FastJsonHttpMessageConverter;
|
||||
import org.springframework.http.MediaType;
|
||||
import org.springframework.http.converter.HttpMessageConverter;
|
||||
|
||||
import java.nio.charset.Charset;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* JSON数据转换器,用于转换返回消息的格式
|
||||
* @author dav
|
||||
* @date 2018/9/11 19:30
|
||||
*/
|
||||
public class JsonConverter {
|
||||
|
||||
/**
|
||||
* FastJson消息转换器
|
||||
*
|
||||
* @return
|
||||
*/
|
||||
public static HttpMessageConverter fastConverter() {
|
||||
// 定义一个convert转换消息的对象
|
||||
FastJsonHttpMessageConverter fastConverter = new FastJsonHttpMessageConverter();
|
||||
// 添加FastJson的配置信息
|
||||
FastJsonConfig fastJsonConfig = new FastJsonConfig();
|
||||
// 默认转换器
|
||||
fastJsonConfig.setSerializerFeatures(SerializerFeature.PrettyFormat,
|
||||
SerializerFeature.WriteNullNumberAsZero,
|
||||
SerializerFeature.MapSortField,
|
||||
SerializerFeature.WriteNullStringAsEmpty,
|
||||
SerializerFeature.DisableCircularReferenceDetect,
|
||||
SerializerFeature.WriteDateUseDateFormat,
|
||||
SerializerFeature.WriteNullListAsEmpty);
|
||||
fastJsonConfig.setCharset(Charset.forName("UTF-8"));
|
||||
// 处理中文乱码问题
|
||||
List<MediaType> fastMediaTypes = new ArrayList<>();
|
||||
fastMediaTypes.add(MediaType.APPLICATION_JSON_UTF8);
|
||||
fastConverter.setSupportedMediaTypes(fastMediaTypes);
|
||||
// 在convert中添加配置信息
|
||||
fastConverter.setFastJsonConfig(fastJsonConfig);
|
||||
|
||||
return fastConverter;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,19 @@
|
|||
package com.guwan.backend.core.enums;
|
||||
|
||||
/**
|
||||
* 通用的状态枚举信息
|
||||
*
|
||||
* @author bool
|
||||
* @date 2019-09-17 17:57
|
||||
*/
|
||||
public interface CommonState {
|
||||
|
||||
/**
|
||||
* 普通状态,正常的
|
||||
*/
|
||||
Integer NORMAL = 0;
|
||||
/**
|
||||
* 非正常状态,禁用,下架等
|
||||
*/
|
||||
Integer ABNORMAL = 1;
|
||||
}
|
|
@ -0,0 +1,18 @@
|
|||
package com.guwan.backend.core.enums;
|
||||
|
||||
/**
|
||||
* 开放方式
|
||||
* @author bool
|
||||
*/
|
||||
public interface OpenType {
|
||||
|
||||
/**
|
||||
* 完全开放
|
||||
*/
|
||||
Integer OPEN = 1;
|
||||
|
||||
/**
|
||||
* 部门开放
|
||||
*/
|
||||
Integer DEPT_OPEN = 2;
|
||||
}
|
|
@ -0,0 +1,52 @@
|
|||
package com.guwan.backend.core.exception;
|
||||
|
||||
|
||||
import com.guwan.backend.core.api.ApiError;
|
||||
import com.guwan.backend.core.api.ApiRest;
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
@Data
|
||||
@AllArgsConstructor
|
||||
@NoArgsConstructor
|
||||
public class ServiceException extends RuntimeException{
|
||||
|
||||
/**
|
||||
* 错误码
|
||||
*/
|
||||
private Integer code;
|
||||
|
||||
/**
|
||||
* 错误消息
|
||||
*/
|
||||
private String msg;
|
||||
|
||||
/**
|
||||
* 从结果初始化
|
||||
* @param apiRest
|
||||
*/
|
||||
public ServiceException(ApiRest apiRest){
|
||||
this.code = apiRest.getCode();
|
||||
this.msg = apiRest.getMsg();
|
||||
}
|
||||
|
||||
/**
|
||||
* 从枚举中获取参数
|
||||
* @param apiError
|
||||
*/
|
||||
public ServiceException(ApiError apiError){
|
||||
this.code = apiError.getCode();
|
||||
this.msg = apiError.msg;
|
||||
}
|
||||
|
||||
/**
|
||||
* 异常构造
|
||||
* @param msg
|
||||
*/
|
||||
public ServiceException(String msg){
|
||||
this.code = 1;
|
||||
this.msg = msg;
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,47 @@
|
|||
package com.guwan.backend.core.exception;
|
||||
|
||||
|
||||
import com.guwan.backend.core.api.ApiRest;
|
||||
import org.springframework.http.HttpStatus;
|
||||
import org.springframework.ui.Model;
|
||||
import org.springframework.web.bind.WebDataBinder;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
|
||||
/**
|
||||
* 统一异常处理类
|
||||
* @author bool
|
||||
* @date 2019-06-21 19:27
|
||||
*/
|
||||
@RestControllerAdvice
|
||||
public class ServiceExceptionHandler {
|
||||
|
||||
/**
|
||||
* 应用到所有@RequestMapping注解方法,在其执行之前初始化数据绑定器
|
||||
* @param binder
|
||||
*/
|
||||
@InitBinder
|
||||
public void initWebBinder(WebDataBinder binder){
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* 把值绑定到Model中,使全局@RequestMapping可以获取到该值
|
||||
* @param model
|
||||
*/
|
||||
@ModelAttribute
|
||||
public void addAttribute(Model model) {
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* 捕获ServiceException
|
||||
* @param e
|
||||
* @return
|
||||
*/
|
||||
@ExceptionHandler({com.guwan.backend.core.exception.ServiceException.class})
|
||||
@ResponseStatus(HttpStatus.OK)
|
||||
public ApiRest serviceExceptionHandler(ServiceException e) {
|
||||
return new ApiRest(e);
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,31 @@
|
|||
package com.guwan.backend.core.utils;
|
||||
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* 时间转换quartz表达式
|
||||
* @author bool
|
||||
* @date 2020/11/29 下午3:00
|
||||
*/
|
||||
public class CronUtils {
|
||||
|
||||
/**
|
||||
* 格式化数据
|
||||
*/
|
||||
private static final String DATE_FORMAT = "ss mm HH dd MM ? yyyy";
|
||||
|
||||
/**
|
||||
* 准确的时间点到表达式
|
||||
* @param date
|
||||
* @return
|
||||
*/
|
||||
public static String dateToCron(final Date date){
|
||||
SimpleDateFormat fmt = new SimpleDateFormat(DATE_FORMAT);
|
||||
String formatTimeStr = "";
|
||||
if (date != null) {
|
||||
formatTimeStr = fmt.format(date);
|
||||
}
|
||||
return formatTimeStr;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,103 @@
|
|||
package com.guwan.backend.core.utils;
|
||||
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Calendar;
|
||||
import java.util.Date;
|
||||
import java.util.GregorianCalendar;
|
||||
|
||||
/**
|
||||
* 日期处理工具类
|
||||
* ClassName: DateUtils <br/>
|
||||
* date: 2018年12月13日 下午6:34:02 <br/>
|
||||
*
|
||||
* @author Bool
|
||||
* @version
|
||||
*/
|
||||
public class DateUtils {
|
||||
|
||||
/**
|
||||
*
|
||||
* calcExpDays:计算某个日期与当前日期相差的天数,如果计算的日期大于现在时间,将返回负数;否则返回正数 <br/>
|
||||
* @author Bool
|
||||
* @param userCreateTime
|
||||
* @return
|
||||
* @since JDK 1.6
|
||||
*/
|
||||
public static int calcExpDays(Date userCreateTime){
|
||||
|
||||
Calendar start = Calendar.getInstance();
|
||||
start.setTime(userCreateTime);
|
||||
|
||||
Calendar now = Calendar.getInstance();
|
||||
now.setTime(new Date());
|
||||
|
||||
long l = now.getTimeInMillis() - start.getTimeInMillis();
|
||||
int days = new Long(l / (1000 * 60 * 60 * 24)).intValue();
|
||||
return days;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* dateNow:获取当前时间的字符串格式,根据传入的格式化来展示. <br/>
|
||||
* @author Bool
|
||||
* @param format 日期格式化
|
||||
* @return
|
||||
*/
|
||||
public static String dateNow(String format) {
|
||||
SimpleDateFormat fmt = new SimpleDateFormat(format);
|
||||
Calendar c = new GregorianCalendar();
|
||||
return fmt.format(c.getTime());
|
||||
}
|
||||
|
||||
/**
|
||||
* formatDate:格式化日期,返回指定的格式 <br/>
|
||||
* @author Bool
|
||||
* @param time
|
||||
* @param format
|
||||
* @return
|
||||
*/
|
||||
public static String formatDate(Date time, String format) {
|
||||
SimpleDateFormat fmt = new SimpleDateFormat(format);
|
||||
return fmt.format(time.getTime());
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* parseDate:将字符串转换成日期,使用:yyyy-MM-dd HH:mm:ss 来格式化
|
||||
* @author Bool
|
||||
* @param date
|
||||
* @return
|
||||
*/
|
||||
public static Date parseDate(String date) {
|
||||
return parseDate(date, "yyyy-MM-dd HH:mm:ss");
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* parseDate:将字符串转换成日期,使用指定格式化来格式化
|
||||
* @author Bool
|
||||
* @param date
|
||||
* @param pattern
|
||||
* @return
|
||||
*/
|
||||
public static Date parseDate(String date, String pattern) {
|
||||
|
||||
if (pattern==null) {
|
||||
pattern = "yyyy-MM-dd HH:mm:ss";
|
||||
}
|
||||
|
||||
SimpleDateFormat fmt = new SimpleDateFormat(pattern);
|
||||
|
||||
try {
|
||||
|
||||
return fmt.parse(date);
|
||||
} catch (Exception ex) {
|
||||
ex.printStackTrace();
|
||||
}
|
||||
return null;
|
||||
|
||||
}
|
||||
}
|
|
@ -0,0 +1,65 @@
|
|||
package com.guwan.backend.core.utils;
|
||||
|
||||
|
||||
import javax.servlet.http.HttpServletRequest;
|
||||
|
||||
/**
|
||||
* IP获取工具类,用户获取网络请求过来的真实IP
|
||||
* ClassName: IpUtils <br/>
|
||||
* date: 2018年2月13日 下午7:27:52 <br/>
|
||||
*
|
||||
* @author Bool
|
||||
* @version
|
||||
*/
|
||||
public class IpUtils {
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* getClientIp:通过请求获取客户端的真实IP地址
|
||||
* @author Bool
|
||||
* @param request
|
||||
* @return
|
||||
*/
|
||||
public static String extractClientIp(HttpServletRequest request) {
|
||||
|
||||
String ip = null;
|
||||
|
||||
//X-Forwarded-For:Squid 服务代理
|
||||
String ipAddresses = request.getHeader("X-Forwarded-For");
|
||||
|
||||
if (ipAddresses == null || ipAddresses.length() == 0 || "unknown".equalsIgnoreCase(ipAddresses)) {
|
||||
//Proxy-Client-IP:apache 服务代理
|
||||
ipAddresses = request.getHeader("Proxy-Client-IP");
|
||||
}
|
||||
|
||||
if (ipAddresses == null || ipAddresses.length() == 0 || "unknown".equalsIgnoreCase(ipAddresses)) {
|
||||
//WL-Proxy-Client-IP:weblogic 服务代理
|
||||
ipAddresses = request.getHeader("WL-Proxy-Client-IP");
|
||||
}
|
||||
|
||||
if (ipAddresses == null || ipAddresses.length() == 0 || "unknown".equalsIgnoreCase(ipAddresses)) {
|
||||
//HTTP_CLIENT_IP:有些代理服务器
|
||||
ipAddresses = request.getHeader("HTTP_CLIENT_IP");
|
||||
}
|
||||
|
||||
if (ipAddresses == null || ipAddresses.length() == 0 || "unknown".equalsIgnoreCase(ipAddresses)) {
|
||||
//X-Real-IP:nginx服务代理
|
||||
ipAddresses = request.getHeader("X-Real-IP");
|
||||
}
|
||||
|
||||
//有些网络通过多层代理,那么获取到的ip就会有多个,一般都是通过逗号(,)分割开来,并且第一个ip为客户端的真实IP
|
||||
if (ipAddresses != null && ipAddresses.length() != 0) {
|
||||
ip = ipAddresses.split(",")[0];
|
||||
}
|
||||
|
||||
//还是不能获取到,最后再通过request.getRemoteAddr();获取
|
||||
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ipAddresses)) {
|
||||
ip = request.getRemoteAddr();
|
||||
}
|
||||
|
||||
return ip;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,319 @@
|
|||
/**
|
||||
* Copyright (c) 2005-2012 springside.org.cn
|
||||
*/
|
||||
package com.guwan.backend.core.utils;
|
||||
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.apache.commons.lang3.Validate;
|
||||
import org.springframework.util.Assert;
|
||||
|
||||
import java.lang.reflect.*;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* 反射工具类.
|
||||
* 提供调用getter/setter方法, 访问私有变量, 调用私有方法, 获取泛型类型Class, 被AOP过的真实类等工具函数.
|
||||
* @author calvin
|
||||
* @version 2016-01-15
|
||||
*/
|
||||
@Log4j2
|
||||
public class Reflections {
|
||||
|
||||
private static final String SETTER_PREFIX = "set";
|
||||
|
||||
private static final String GETTER_PREFIX = "get";
|
||||
|
||||
private static final String CGLIB_CLASS_SEPARATOR = "$$";
|
||||
|
||||
|
||||
/**
|
||||
* 获取类的所有属性,包括父类
|
||||
*
|
||||
* @param object
|
||||
* @return
|
||||
*/
|
||||
public static Field[] getAllFields(Object object) {
|
||||
Class<?> clazz = object.getClass();
|
||||
List<Field> fieldList = new ArrayList<>();
|
||||
while (clazz != null) {
|
||||
fieldList.addAll(new ArrayList<>(Arrays.asList(clazz.getDeclaredFields())));
|
||||
clazz = clazz.getSuperclass();
|
||||
}
|
||||
Field[] fields = new Field[fieldList.size()];
|
||||
fieldList.toArray(fields);
|
||||
return fields;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 调用Getter方法.
|
||||
* 支持多级,如:对象名.对象名.方法
|
||||
*/
|
||||
public static Object invokeGetter(Object obj, String propertyName) {
|
||||
Object object = obj;
|
||||
for (String name : StringUtils.split(propertyName, ".")){
|
||||
String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(name);
|
||||
object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {});
|
||||
}
|
||||
return object;
|
||||
}
|
||||
|
||||
/**
|
||||
* 调用Setter方法, 仅匹配方法名。
|
||||
* 支持多级,如:对象名.对象名.方法
|
||||
*/
|
||||
public static void invokeSetter(Object obj, String propertyName, Object value) {
|
||||
Object object = obj;
|
||||
String[] names = StringUtils.split(propertyName, ".");
|
||||
for (int i=0; i<names.length; i++){
|
||||
if(i<names.length-1){
|
||||
String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(names[i]);
|
||||
object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {});
|
||||
}else{
|
||||
String setterMethodName = SETTER_PREFIX + StringUtils.capitalize(names[i]);
|
||||
invokeMethodByName(object, setterMethodName, new Object[] { value });
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 直接读取对象属性值, 无视private/protected修饰符, 不经过getter函数.
|
||||
*/
|
||||
public static Object getFieldValue(final Object obj, final String fieldName) {
|
||||
Field field = getAccessibleField(obj, fieldName);
|
||||
|
||||
if (field == null) {
|
||||
throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + obj + "]");
|
||||
}
|
||||
|
||||
Object result = null;
|
||||
try {
|
||||
result = field.get(obj);
|
||||
} catch (IllegalAccessException e) {
|
||||
log.error("不可能抛出的异常{}", e.getMessage());
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 直接设置对象属性值, 无视private/protected修饰符, 不经过setter函数.
|
||||
*/
|
||||
public static void setFieldValue(final Object obj, final String fieldName, final Object value) {
|
||||
Field field = getAccessibleField(obj, fieldName);
|
||||
|
||||
if (field == null) {
|
||||
throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + obj + "]");
|
||||
}
|
||||
|
||||
try {
|
||||
field.set(obj, value);
|
||||
} catch (IllegalAccessException e) {
|
||||
log.error("不可能抛出的异常:{}", e.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 直接调用对象方法, 无视private/protected修饰符.
|
||||
* 用于一次性调用的情况,否则应使用getAccessibleMethod()函数获得Method后反复调用.
|
||||
* 同时匹配方法名+参数类型,
|
||||
*/
|
||||
public static Object invokeMethod(final Object obj, final String methodName, final Class<?>[] parameterTypes,
|
||||
final Object[] args) {
|
||||
Method method = getAccessibleMethod(obj, methodName, parameterTypes);
|
||||
if (method == null) {
|
||||
throw new IllegalArgumentException("Could not find method [" + methodName + "] on target [" + obj + "]");
|
||||
}
|
||||
|
||||
try {
|
||||
return method.invoke(obj, args);
|
||||
} catch (Exception e) {
|
||||
throw convertReflectionExceptionToUnchecked(e);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 直接调用对象方法, 无视private/protected修饰符,
|
||||
* 用于一次性调用的情况,否则应使用getAccessibleMethodByName()函数获得Method后反复调用.
|
||||
* 只匹配函数名,如果有多个同名函数调用第一个。
|
||||
*/
|
||||
public static Object invokeMethodByName(final Object obj, final String methodName, final Object[] args) {
|
||||
Method method = getAccessibleMethodByName(obj, methodName);
|
||||
if (method == null) {
|
||||
throw new IllegalArgumentException("Could not find method [" + methodName + "] on target [" + obj + "]");
|
||||
}
|
||||
|
||||
try {
|
||||
return method.invoke(obj, args);
|
||||
} catch (Exception e) {
|
||||
throw convertReflectionExceptionToUnchecked(e);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 循环向上转型, 获取对象的DeclaredField, 并强制设置为可访问.
|
||||
*
|
||||
* 如向上转型到Object仍无法找到, 返回null.
|
||||
*/
|
||||
public static Field getAccessibleField(final Object obj, final String fieldName) {
|
||||
Validate.notNull(obj, "object can't be null");
|
||||
Validate.notBlank(fieldName, "fieldName can't be blank");
|
||||
for (Class<?> superClass = obj.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()) {
|
||||
try {
|
||||
Field field = superClass.getDeclaredField(fieldName);
|
||||
makeAccessible(field);
|
||||
return field;
|
||||
} catch (NoSuchFieldException e) {//NOSONAR
|
||||
// Field不在当前类定义,继续向上转型
|
||||
continue;// new add
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* 循环向上转型, 获取对象的DeclaredMethod,并强制设置为可访问.
|
||||
* 如向上转型到Object仍无法找到, 返回null.
|
||||
* 匹配函数名+参数类型。
|
||||
*
|
||||
* 用于方法需要被多次调用的情况. 先使用本函数先取得Method,然后调用Method.invoke(Object obj, Object... args)
|
||||
*/
|
||||
public static Method getAccessibleMethod(final Object obj, final String methodName,
|
||||
final Class<?>... parameterTypes) {
|
||||
Validate.notNull(obj, "object can't be null");
|
||||
Validate.notBlank(methodName, "methodName can't be blank");
|
||||
|
||||
for (Class<?> searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) {
|
||||
try {
|
||||
Method method = searchType.getDeclaredMethod(methodName, parameterTypes);
|
||||
makeAccessible(method);
|
||||
return method;
|
||||
} catch (NoSuchMethodException e) {
|
||||
// Method不在当前类定义,继续向上转型
|
||||
continue;// new add
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* 循环向上转型, 获取对象的DeclaredMethod,并强制设置为可访问.
|
||||
* 如向上转型到Object仍无法找到, 返回null.
|
||||
* 只匹配函数名。
|
||||
*
|
||||
* 用于方法需要被多次调用的情况. 先使用本函数先取得Method,然后调用Method.invoke(Object obj, Object... args)
|
||||
*/
|
||||
public static Method getAccessibleMethodByName(final Object obj, final String methodName) {
|
||||
Validate.notNull(obj, "object can't be null");
|
||||
Validate.notBlank(methodName, "methodName can't be blank");
|
||||
|
||||
for (Class<?> searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) {
|
||||
Method[] methods = searchType.getDeclaredMethods();
|
||||
for (Method method : methods) {
|
||||
if (method.getName().equals(methodName)) {
|
||||
makeAccessible(method);
|
||||
return method;
|
||||
}
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* 改变private/protected的方法为public,尽量不调用实际改动的语句,避免JDK的SecurityManager抱怨。
|
||||
*/
|
||||
public static void makeAccessible(Method method) {
|
||||
if ((!Modifier.isPublic(method.getModifiers()) || !Modifier.isPublic(method.getDeclaringClass().getModifiers()))
|
||||
&& !method.isAccessible()) {
|
||||
method.setAccessible(true);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 改变private/protected的成员变量为public,尽量不调用实际改动的语句,避免JDK的SecurityManager抱怨。
|
||||
*/
|
||||
public static void makeAccessible(Field field) {
|
||||
if ((!Modifier.isPublic(field.getModifiers()) || !Modifier.isPublic(field.getDeclaringClass().getModifiers()) || Modifier
|
||||
.isFinal(field.getModifiers())) && !field.isAccessible()) {
|
||||
field.setAccessible(true);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 通过反射, 获得Class定义中声明的泛型参数的类型, 注意泛型必须定义在父类处
|
||||
* 如无法找到, 返回Object.class.
|
||||
* eg.
|
||||
* public UserDao extends HibernateDao<User>
|
||||
*
|
||||
* @param clazz The class to introspect
|
||||
* @return the first generic declaration, or Object.class if cannot be determined
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
public static <T> Class<T> getClassGenricType(final Class clazz) {
|
||||
return getClassGenricType(clazz, 0);
|
||||
}
|
||||
|
||||
/**
|
||||
* 通过反射, 获得Class定义中声明的父类的泛型参数的类型.
|
||||
* 如无法找到, 返回Object.class.
|
||||
*
|
||||
* 如public UserDao extends HibernateDao<User,Long>
|
||||
*
|
||||
* @param clazz clazz The class to introspect
|
||||
* @param index the Index of the generic ddeclaration,start from 0.
|
||||
* @return the index generic declaration, or Object.class if cannot be determined
|
||||
*/
|
||||
public static Class getClassGenricType(final Class clazz, final int index) {
|
||||
|
||||
Type genType = clazz.getGenericSuperclass();
|
||||
|
||||
if (!(genType instanceof ParameterizedType)) {
|
||||
log.warn(clazz.getSimpleName() + "'s superclass not ParameterizedType");
|
||||
return Object.class;
|
||||
}
|
||||
|
||||
Type[] params = ((ParameterizedType) genType).getActualTypeArguments();
|
||||
|
||||
if (index >= params.length || index < 0) {
|
||||
log.warn("Index: " + index + ", Size of " + clazz.getSimpleName() + "'s Parameterized Type: "
|
||||
+ params.length);
|
||||
return Object.class;
|
||||
}
|
||||
if (!(params[index] instanceof Class)) {
|
||||
log.warn(clazz.getSimpleName() + " not set the actual class on superclass generic parameter");
|
||||
return Object.class;
|
||||
}
|
||||
|
||||
return (Class) params[index];
|
||||
}
|
||||
|
||||
public static Class<?> getUserClass(Object instance) {
|
||||
Assert.notNull(instance, "Instance must not be null");
|
||||
Class clazz = instance.getClass();
|
||||
if (clazz != null && clazz.getName().contains(CGLIB_CLASS_SEPARATOR)) {
|
||||
Class<?> superClass = clazz.getSuperclass();
|
||||
if (superClass != null && !Object.class.equals(superClass)) {
|
||||
return superClass;
|
||||
}
|
||||
}
|
||||
return clazz;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* 将反射时的checked exception转换为unchecked exception.
|
||||
*/
|
||||
public static RuntimeException convertReflectionExceptionToUnchecked(Exception e) {
|
||||
if (e instanceof IllegalAccessException || e instanceof IllegalArgumentException
|
||||
|| e instanceof NoSuchMethodException) {
|
||||
return new IllegalArgumentException(e);
|
||||
} else if (e instanceof InvocationTargetException) {
|
||||
return new RuntimeException(((InvocationTargetException) e).getTargetException());
|
||||
} else if (e instanceof RuntimeException) {
|
||||
return (RuntimeException) e;
|
||||
}
|
||||
return new RuntimeException("Unexpected Checked Exception.", e);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,32 @@
|
|||
package com.guwan.backend.core.utils;
|
||||
|
||||
import org.springframework.beans.BeansException;
|
||||
import org.springframework.context.ApplicationContext;
|
||||
import org.springframework.context.ApplicationContextAware;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
/**
|
||||
* Spring获取工具
|
||||
*
|
||||
* @author bool
|
||||
* @date 2019-12-09 15:55
|
||||
*/
|
||||
@Component
|
||||
public class SpringUtils implements ApplicationContextAware {
|
||||
|
||||
private static ApplicationContext applicationContext;
|
||||
|
||||
@Override
|
||||
public void setApplicationContext(ApplicationContext context) throws BeansException {
|
||||
applicationContext = context;
|
||||
}
|
||||
|
||||
public static <T> T getBean(Class<T> tClass) {
|
||||
return applicationContext.getBean(tClass);
|
||||
}
|
||||
|
||||
public static <T> T getBean(String name, Class<T> type) {
|
||||
return applicationContext.getBean(name, type);
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
package com.guwan.backend.core.utils;
|
||||
|
||||
import java.util.Map;
|
||||
|
||||
/**
|
||||
* 字符串常用工具类
|
||||
* @author bool
|
||||
* @date 2019-05-15 11:40
|
||||
*/
|
||||
public class StringUtils {
|
||||
|
||||
/**
|
||||
* 判断是否为空字符
|
||||
* @param str
|
||||
* @return
|
||||
*/
|
||||
public static boolean isBlank(String str){
|
||||
return str==null || "".equals(str);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 将MAP转换成一个xml格式,格式为<xml><key>value</key>...</xml>
|
||||
* @param params
|
||||
* @return
|
||||
*/
|
||||
public static String mapToXml(Map<String, String> params){
|
||||
StringBuffer sb = new StringBuffer("<xml>");
|
||||
for(String key:params.keySet()){
|
||||
sb.append("<")
|
||||
.append(key).append(">")
|
||||
.append(params.get(key))
|
||||
.append("</").append(key).append(">");
|
||||
}
|
||||
|
||||
sb.append("</xml>");
|
||||
return sb.toString();
|
||||
}
|
||||
}
|
|
@ -0,0 +1,391 @@
|
|||
/**
|
||||
* Copyright © 2015-2020 <a href="http://www.jeeplus.org/">JeePlus</a> All rights reserved.
|
||||
*/
|
||||
package com.guwan.backend.core.utils.excel;
|
||||
|
||||
import com.google.common.collect.Lists;
|
||||
|
||||
import com.guwan.backend.core.utils.Reflections;
|
||||
import com.guwan.backend.core.utils.excel.annotation.ExcelField;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.apache.poi.ss.usermodel.*;
|
||||
import org.apache.poi.ss.util.CellRangeAddress;
|
||||
import org.apache.poi.xssf.streaming.SXSSFWorkbook;
|
||||
import org.apache.poi.xssf.usermodel.XSSFClientAnchor;
|
||||
import org.apache.poi.xssf.usermodel.XSSFRichTextString;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import javax.servlet.http.HttpServletResponse;
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.Method;
|
||||
import java.net.URLEncoder;
|
||||
import java.util.*;
|
||||
|
||||
/**
|
||||
* 导出Excel文件(导出“XLSX”格式,支持大数据量导出 @see org.apache.poi.ss.SpreadsheetVersion)
|
||||
* @author jeeplus
|
||||
* @version 2016-04-21
|
||||
*/
|
||||
public class ExportExcel {
|
||||
|
||||
private static Logger log = LoggerFactory.getLogger(ExportExcel.class);
|
||||
|
||||
/**
|
||||
* 工作薄对象
|
||||
*/
|
||||
private SXSSFWorkbook wb;
|
||||
|
||||
/**
|
||||
* 工作表对象
|
||||
*/
|
||||
private Sheet sheet;
|
||||
|
||||
/**
|
||||
* 样式列表
|
||||
*/
|
||||
private Map<String, CellStyle> styles;
|
||||
|
||||
/**
|
||||
* 当前行号
|
||||
*/
|
||||
private int rownum;
|
||||
|
||||
/**
|
||||
* 注解列表(Object[]{ ExcelField, Field/Method })
|
||||
*/
|
||||
List<Object[]> annotationList = Lists.newArrayList();
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
* @param title 表格标题,传“空值”,表示无标题
|
||||
* @param cls 实体对象,通过annotation.ExportField获取标题
|
||||
*/
|
||||
public ExportExcel(String title, Class<?> cls){
|
||||
this(title, cls, 1);
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
* @param title 表格标题,传“空值”,表示无标题
|
||||
* @param cls 实体对象,通过annotation.ExportField获取标题
|
||||
* @param type 导出类型(1:导出数据;2:导出模板)
|
||||
* @param groups 导入分组
|
||||
*/
|
||||
public ExportExcel(String title, Class<?> cls, int type, int... groups){
|
||||
// Get annotation field
|
||||
Field[] fs = cls.getDeclaredFields();
|
||||
for (Field f : fs){
|
||||
ExcelField ef = f.getAnnotation(ExcelField.class);
|
||||
if (ef != null && (ef.type()==0 || ef.type()==type)){
|
||||
if (groups!=null && groups.length>0){
|
||||
boolean inGroup = false;
|
||||
for (int g : groups){
|
||||
if (inGroup){
|
||||
break;
|
||||
}
|
||||
for (int efg : ef.groups()){
|
||||
if (g == efg){
|
||||
inGroup = true;
|
||||
annotationList.add(new Object[]{ef, f});
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}else{
|
||||
annotationList.add(new Object[]{ef, f});
|
||||
}
|
||||
}
|
||||
}
|
||||
// Get annotation method
|
||||
Method[] ms = cls.getDeclaredMethods();
|
||||
for (Method m : ms){
|
||||
ExcelField ef = m.getAnnotation(ExcelField.class);
|
||||
if (ef != null && (ef.type()==0 || ef.type()==type)){
|
||||
if (groups!=null && groups.length>0){
|
||||
boolean inGroup = false;
|
||||
for (int g : groups){
|
||||
if (inGroup){
|
||||
break;
|
||||
}
|
||||
for (int efg : ef.groups()){
|
||||
if (g == efg){
|
||||
inGroup = true;
|
||||
annotationList.add(new Object[]{ef, m});
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}else{
|
||||
annotationList.add(new Object[]{ef, m});
|
||||
}
|
||||
}
|
||||
}
|
||||
// Field sorting
|
||||
Collections.sort(annotationList, new Comparator<Object[]>() {
|
||||
@Override
|
||||
public int compare(Object[] o1, Object[] o2) {
|
||||
/*return new Integer(((ExcelField)o1[0]).sort()).compareTo(
|
||||
new Integer(((ExcelField)o2[0]).sort()));*/
|
||||
return Integer.compare(((ExcelField) o1[0]).sort(), ((ExcelField) o2[0]).sort());
|
||||
}
|
||||
});
|
||||
// Initialize
|
||||
List<String> headerList = Lists.newArrayList();
|
||||
for (Object[] os : annotationList){
|
||||
String t = ((ExcelField)os[0]).title();
|
||||
// 如果是导出,则去掉注释
|
||||
if (type==1){
|
||||
String[] ss = StringUtils.split(t, "**", 2);
|
||||
if (ss.length==2){
|
||||
t = ss[0];
|
||||
}
|
||||
}
|
||||
headerList.add(t);
|
||||
}
|
||||
initialize(title, headerList);
|
||||
}
|
||||
/**
|
||||
* 初始化函数
|
||||
* @param title 表格标题,传“空值”,表示无标题
|
||||
* @param headerList 表头列表
|
||||
*/
|
||||
private void initialize(String title, List<String> headerList) {
|
||||
this.wb = new SXSSFWorkbook(500);
|
||||
this.sheet = wb.createSheet("Export");
|
||||
this.styles = createStyles(wb);
|
||||
// Create title
|
||||
if (StringUtils.isNotBlank(title)){
|
||||
Row titleRow = sheet.createRow(rownum++);
|
||||
titleRow.setHeightInPoints(30);
|
||||
Cell titleCell = titleRow.createCell(0);
|
||||
titleCell.setCellStyle(styles.get("title"));
|
||||
titleCell.setCellValue(title);
|
||||
sheet.addMergedRegion(new CellRangeAddress(titleRow.getRowNum(),
|
||||
titleRow.getRowNum(), titleRow.getRowNum(), headerList.size()-1));
|
||||
}
|
||||
// Create header
|
||||
if (headerList == null){
|
||||
throw new RuntimeException("headerList not null!");
|
||||
}
|
||||
Row headerRow = sheet.createRow(rownum++);
|
||||
headerRow.setHeightInPoints(16);
|
||||
for (int i = 0; i < headerList.size(); i++) {
|
||||
Cell cell = headerRow.createCell(i);
|
||||
cell.setCellStyle(styles.get("header"));
|
||||
String[] ss = StringUtils.split(headerList.get(i), "**", 2);
|
||||
if (ss.length==2){
|
||||
cell.setCellValue(ss[0]);
|
||||
Comment comment = this.sheet.createDrawingPatriarch().createCellComment(
|
||||
new XSSFClientAnchor(0, 0, 0, 0, (short) 3, 3, (short) 5, 6));
|
||||
comment.setString(new XSSFRichTextString(ss[1]));
|
||||
cell.setCellComment(comment);
|
||||
}else{
|
||||
cell.setCellValue(headerList.get(i));
|
||||
}
|
||||
sheet.autoSizeColumn(i);
|
||||
}
|
||||
for (int i = 0; i < headerList.size(); i++) {
|
||||
int colWidth = sheet.getColumnWidth(i)*2;
|
||||
sheet.setColumnWidth(i, colWidth < 3000 ? 3000 : colWidth);
|
||||
}
|
||||
log.debug("Initialize success.");
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建表格样式
|
||||
* @param wb 工作薄对象
|
||||
* @return 样式列表
|
||||
*/
|
||||
private Map<String, CellStyle> createStyles(Workbook wb) {
|
||||
Map<String, CellStyle> styles = new HashMap<>(16);
|
||||
|
||||
CellStyle style = wb.createCellStyle();
|
||||
style.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
style.setVerticalAlignment(CellStyle.VERTICAL_CENTER);
|
||||
Font titleFont = wb.createFont();
|
||||
titleFont.setFontName("Arial");
|
||||
titleFont.setFontHeightInPoints((short) 16);
|
||||
titleFont.setBoldweight(Font.BOLDWEIGHT_BOLD);
|
||||
style.setFont(titleFont);
|
||||
styles.put("title", style);
|
||||
|
||||
style = wb.createCellStyle();
|
||||
style.setVerticalAlignment(CellStyle.VERTICAL_CENTER);
|
||||
style.setBorderRight(CellStyle.BORDER_THIN);
|
||||
style.setRightBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
|
||||
style.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
style.setLeftBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
|
||||
style.setBorderTop(CellStyle.BORDER_THIN);
|
||||
style.setTopBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
|
||||
style.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
style.setBottomBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
|
||||
Font dataFont = wb.createFont();
|
||||
dataFont.setFontName("Arial");
|
||||
dataFont.setFontHeightInPoints((short) 10);
|
||||
style.setFont(dataFont);
|
||||
styles.put("data", style);
|
||||
|
||||
style = wb.createCellStyle();
|
||||
style.cloneStyleFrom(styles.get("data"));
|
||||
style.setAlignment(CellStyle.ALIGN_LEFT);
|
||||
styles.put("data1", style);
|
||||
|
||||
style = wb.createCellStyle();
|
||||
style.cloneStyleFrom(styles.get("data"));
|
||||
style.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
styles.put("data2", style);
|
||||
|
||||
style = wb.createCellStyle();
|
||||
style.cloneStyleFrom(styles.get("data"));
|
||||
style.setAlignment(CellStyle.ALIGN_RIGHT);
|
||||
styles.put("data3", style);
|
||||
|
||||
style = wb.createCellStyle();
|
||||
style.cloneStyleFrom(styles.get("data"));
|
||||
// style.setWrapText(true);
|
||||
style.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
style.setFillForegroundColor(IndexedColors.GREY_50_PERCENT.getIndex());
|
||||
style.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
Font headerFont = wb.createFont();
|
||||
headerFont.setFontName("Arial");
|
||||
headerFont.setFontHeightInPoints((short) 10);
|
||||
headerFont.setBoldweight(Font.BOLDWEIGHT_BOLD);
|
||||
headerFont.setColor(IndexedColors.WHITE.getIndex());
|
||||
style.setFont(headerFont);
|
||||
styles.put("header", style);
|
||||
|
||||
return styles;
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加一行
|
||||
* @return 行对象
|
||||
*/
|
||||
public Row addRow(){
|
||||
return sheet.createRow(rownum++);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 添加一个单元格
|
||||
* @param row 添加的行
|
||||
* @param column 添加列号
|
||||
* @param val 添加值
|
||||
* @return 单元格对象
|
||||
*/
|
||||
public Cell addCell(Row row, int column, Object val){
|
||||
return this.addCell(row, column, val, 0, Class.class);
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加一个单元格
|
||||
* @param row 添加的行
|
||||
* @param column 添加列号
|
||||
* @param val 添加值
|
||||
* @param align 对齐方式(1:靠左;2:居中;3:靠右)
|
||||
* @return 单元格对象
|
||||
*/
|
||||
public Cell addCell(Row row, int column, Object val, int align, Class<?> fieldType){
|
||||
Cell cell = row.createCell(column);
|
||||
CellStyle style = styles.get("data"+(align>=1&&align<=3?align:""));
|
||||
try {
|
||||
if (val == null){
|
||||
cell.setCellValue("");
|
||||
} else if (val instanceof String) {
|
||||
cell.setCellValue((String) val);
|
||||
} else if (val instanceof Integer) {
|
||||
cell.setCellValue((Integer) val);
|
||||
} else if (val instanceof Long) {
|
||||
cell.setCellValue((Long) val);
|
||||
} else if (val instanceof Double) {
|
||||
cell.setCellValue((Double) val);
|
||||
} else if (val instanceof Float) {
|
||||
cell.setCellValue((Float) val);
|
||||
} else if (val instanceof Date) {
|
||||
DataFormat format = wb.createDataFormat();
|
||||
style.setDataFormat(format.getFormat("yyyy-MM-dd"));
|
||||
cell.setCellValue((Date) val);
|
||||
} else {
|
||||
if (fieldType != Class.class){
|
||||
cell.setCellValue((String)fieldType.getMethod("setValue", Object.class).invoke(null, val));
|
||||
}else{
|
||||
cell.setCellValue((String)Class.forName(this.getClass().getName().replaceAll(this.getClass().getSimpleName(),
|
||||
"fieldtype."+val.getClass().getSimpleName()+"Type")).getMethod("setValue", Object.class).invoke(null, val));
|
||||
}
|
||||
}
|
||||
} catch (Exception ex) {
|
||||
log.info("Set cell value ["+row.getRowNum()+","+column+"] error: " + ex.toString());
|
||||
cell.setCellValue(val.toString());
|
||||
}
|
||||
cell.setCellStyle(style);
|
||||
return cell;
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加数据(通过annotation.ExportField添加数据)
|
||||
* @return list 数据列表
|
||||
*/
|
||||
public <E> ExportExcel setDataList(List<E> list){
|
||||
for (E e : list){
|
||||
int colunm = 0;
|
||||
Row row = this.addRow();
|
||||
StringBuilder sb = new StringBuilder();
|
||||
for (Object[] os : annotationList){
|
||||
ExcelField ef = (ExcelField)os[0];
|
||||
Object val = null;
|
||||
try{
|
||||
if (StringUtils.isNotBlank(ef.value())){
|
||||
val = Reflections.invokeGetter(e, ef.value());
|
||||
}else{
|
||||
if (os[1] instanceof Field){
|
||||
val = Reflections.invokeGetter(e, ((Field)os[1]).getName());
|
||||
}else if (os[1] instanceof Method){
|
||||
val = Reflections.invokeMethod(e, ((Method)os[1]).getName(), new Class[] {}, new Object[] {});
|
||||
}
|
||||
}
|
||||
}catch(Exception ex) {
|
||||
log.info(ex.toString());
|
||||
val = "";
|
||||
}
|
||||
this.addCell(row, colunm++, val, ef.align(), ef.fieldType());
|
||||
sb.append(val + ", ");
|
||||
}
|
||||
log.debug("Write success: ["+row.getRowNum()+"] "+sb.toString());
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 输出数据流
|
||||
* @param os 输出数据流
|
||||
*/
|
||||
public ExportExcel write(OutputStream os) throws IOException{
|
||||
wb.write(os);
|
||||
return this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 输出到客户端
|
||||
* @param fileName 输出文件名
|
||||
*/
|
||||
public ExportExcel write(HttpServletResponse response, String fileName) throws IOException{
|
||||
response.reset();
|
||||
response.setHeader("Access-Control-Allow-Origin", "*");
|
||||
response.setContentType("application/octet-stream; charset=utf-8");
|
||||
response.addHeader("Content-Disposition", "attachment; filename="+ URLEncoder.encode(fileName, "utf-8"));
|
||||
write(response.getOutputStream());
|
||||
return this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理临时文件
|
||||
*/
|
||||
public ExportExcel dispose(){
|
||||
wb.dispose();
|
||||
return this;
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,305 @@
|
|||
/**
|
||||
* Copyright © 2015-2020 <a href="http://www.jeeplus.org/">JeePlus</a> All rights reserved.
|
||||
*/
|
||||
package com.guwan.backend.core.utils.excel;
|
||||
|
||||
import com.google.common.collect.Lists;
|
||||
|
||||
import com.guwan.backend.core.utils.Reflections;
|
||||
import com.guwan.backend.core.utils.excel.annotation.ExcelField;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.apache.poi.hssf.usermodel.HSSFDateUtil;
|
||||
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
|
||||
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
|
||||
import org.apache.poi.ss.usermodel.Cell;
|
||||
import org.apache.poi.ss.usermodel.Row;
|
||||
import org.apache.poi.ss.usermodel.Sheet;
|
||||
import org.apache.poi.ss.usermodel.Workbook;
|
||||
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
import org.springframework.web.multipart.MultipartFile;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.Method;
|
||||
import java.text.NumberFormat;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Collections;
|
||||
import java.util.Comparator;
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* 导入Excel文件(支持“XLS”和“XLSX”格式)
|
||||
* @author jeeplus
|
||||
* @version 2016-03-10
|
||||
*/
|
||||
public class ImportExcel {
|
||||
|
||||
private static Logger log = LoggerFactory.getLogger(ImportExcel.class);
|
||||
|
||||
/**
|
||||
* 工作薄对象
|
||||
*/
|
||||
private Workbook wb;
|
||||
|
||||
/**
|
||||
* 工作表对象
|
||||
*/
|
||||
private Sheet sheet;
|
||||
|
||||
/**
|
||||
* 标题行号
|
||||
*/
|
||||
private int headerNum;
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
* @param multipartFile 导入文件对象
|
||||
* @param headerNum 标题行号,数据行号=标题行号+1
|
||||
* @param sheetIndex 工作表编号
|
||||
* @throws InvalidFormatException
|
||||
* @throws IOException
|
||||
*/
|
||||
public ImportExcel(MultipartFile multipartFile, int headerNum, int sheetIndex)
|
||||
throws InvalidFormatException, IOException {
|
||||
this(multipartFile.getOriginalFilename(), multipartFile.getInputStream(), headerNum, sheetIndex);
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
* @param is 导入文件对象
|
||||
* @param headerNum 标题行号,数据行号=标题行号+1
|
||||
* @param sheetIndex 工作表编号
|
||||
* @throws InvalidFormatException
|
||||
* @throws IOException
|
||||
*/
|
||||
public ImportExcel(String fileName, InputStream is, int headerNum, int sheetIndex)
|
||||
throws IOException {
|
||||
if (StringUtils.isBlank(fileName)){
|
||||
throw new RuntimeException("导入文档为空!");
|
||||
}else if(fileName.toLowerCase().endsWith("xls")){
|
||||
this.wb = new HSSFWorkbook(is);
|
||||
}else if(fileName.toLowerCase().endsWith("xlsx")){
|
||||
this.wb = new XSSFWorkbook(is);
|
||||
}else{
|
||||
throw new RuntimeException("文档格式不正确!");
|
||||
}
|
||||
if (this.wb.getNumberOfSheets()<sheetIndex){
|
||||
throw new RuntimeException("文档中没有工作表!");
|
||||
}
|
||||
this.sheet = this.wb.getSheetAt(sheetIndex);
|
||||
this.headerNum = headerNum;
|
||||
log.debug("Initialize success.");
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取行对象
|
||||
* @param rownum
|
||||
* @return
|
||||
*/
|
||||
public Row getRow(int rownum){
|
||||
return this.sheet.getRow(rownum);
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取数据行号
|
||||
* @return
|
||||
*/
|
||||
public int getDataRowNum(){
|
||||
return headerNum+1;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取最后一个数据行号
|
||||
* @return
|
||||
*/
|
||||
public int getLastDataRowNum(){
|
||||
return this.sheet.getLastRowNum()+headerNum;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 获取单元格值
|
||||
* @param row 获取的行
|
||||
* @param column 获取单元格列号
|
||||
* @return 单元格值
|
||||
*/
|
||||
public Object getCellValue(Row row, int column) {
|
||||
Object val = "";
|
||||
try {
|
||||
Cell cell = row.getCell(column);
|
||||
if (cell != null) {
|
||||
if (cell.getCellType() == Cell.CELL_TYPE_NUMERIC) {
|
||||
// 当excel 中的数据为数值或日期是需要特殊处理
|
||||
if (HSSFDateUtil.isCellDateFormatted(cell)) {
|
||||
double d = cell.getNumericCellValue();
|
||||
Date date = HSSFDateUtil.getJavaDate(d);
|
||||
SimpleDateFormat dformat = new SimpleDateFormat(
|
||||
"yyyy-MM-dd");
|
||||
val = dformat.format(date);
|
||||
} else {
|
||||
NumberFormat nf = NumberFormat.getInstance();
|
||||
nf.setGroupingUsed(false);// true时的格式:1,234,567,890
|
||||
val = nf.format(cell.getNumericCellValue());// 数值类型的数据为double,所以需要转换一下
|
||||
}
|
||||
} else if (cell.getCellType() == Cell.CELL_TYPE_STRING) {
|
||||
val = cell.getStringCellValue();
|
||||
} else if (cell.getCellType() == Cell.CELL_TYPE_FORMULA) {
|
||||
val = cell.getCellFormula();
|
||||
} else if (cell.getCellType() == Cell.CELL_TYPE_BOOLEAN) {
|
||||
val = cell.getBooleanCellValue();
|
||||
} else if (cell.getCellType() == Cell.CELL_TYPE_ERROR) {
|
||||
val = cell.getErrorCellValue();
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
return val;
|
||||
}
|
||||
return val;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取导入数据列表
|
||||
* @param cls 导入对象类型
|
||||
* @param groups 导入分组
|
||||
*/
|
||||
public <E> List<E> getDataList(Class<E> cls, int... groups) throws InstantiationException, IllegalAccessException{
|
||||
List<Object[]> annotationList = Lists.newArrayList();
|
||||
// Get annotation field
|
||||
Field[] fs = cls.getDeclaredFields();
|
||||
for (Field f : fs){
|
||||
ExcelField ef = f.getAnnotation(ExcelField.class);
|
||||
if (ef != null && (ef.type()==0 || ef.type()==2)){
|
||||
if (groups!=null && groups.length>0){
|
||||
boolean inGroup = false;
|
||||
for (int g : groups){
|
||||
if (inGroup){
|
||||
break;
|
||||
}
|
||||
for (int efg : ef.groups()){
|
||||
if (g == efg){
|
||||
inGroup = true;
|
||||
annotationList.add(new Object[]{ef, f});
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}else{
|
||||
annotationList.add(new Object[]{ef, f});
|
||||
}
|
||||
}
|
||||
}
|
||||
// Get annotation method
|
||||
Method[] ms = cls.getDeclaredMethods();
|
||||
for (Method m : ms){
|
||||
ExcelField ef = m.getAnnotation(ExcelField.class);
|
||||
if (ef != null && (ef.type()==0 || ef.type()==2)){
|
||||
if (groups!=null && groups.length>0){
|
||||
boolean inGroup = false;
|
||||
for (int g : groups){
|
||||
if (inGroup){
|
||||
break;
|
||||
}
|
||||
for (int efg : ef.groups()){
|
||||
if (g == efg){
|
||||
inGroup = true;
|
||||
annotationList.add(new Object[]{ef, m});
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}else{
|
||||
annotationList.add(new Object[]{ef, m});
|
||||
}
|
||||
}
|
||||
}
|
||||
// Field sorting
|
||||
Collections.sort(annotationList, new Comparator<Object[]>() {
|
||||
@Override
|
||||
public int compare(Object[] o1, Object[] o2) {
|
||||
// return new Integer(((ExcelField)o1[0]).sort()).compareTo(
|
||||
// new Integer(((ExcelField)o2[0]).sort()));
|
||||
return Integer.compare(((ExcelField) o1[0]).sort(), ((ExcelField) o2[0]).sort());
|
||||
}
|
||||
});
|
||||
// Get excel data
|
||||
List<E> dataList = Lists.newArrayList();
|
||||
for (int i = this.getDataRowNum(); i < this.getLastDataRowNum(); i++) {
|
||||
E e = (E)cls.newInstance();
|
||||
int column = 0;
|
||||
Row row = this.getRow(i);
|
||||
StringBuilder sb = new StringBuilder();
|
||||
for (Object[] os : annotationList){
|
||||
Object val = this.getCellValue(row, column++);
|
||||
if (val != null){
|
||||
ExcelField ef = (ExcelField)os[0];
|
||||
// Get param type and type cast
|
||||
Class<?> valType = Class.class;
|
||||
if (os[1] instanceof Field){
|
||||
valType = ((Field)os[1]).getType();
|
||||
}else if (os[1] instanceof Method){
|
||||
Method method = ((Method)os[1]);
|
||||
if ("get".equals(method.getName().substring(0, 3))){
|
||||
valType = method.getReturnType();
|
||||
}else if("set".equals(method.getName().substring(0, 3))){
|
||||
valType = ((Method)os[1]).getParameterTypes()[0];
|
||||
}
|
||||
}
|
||||
//log.debug("Import value type: ["+i+","+column+"] " + valType);
|
||||
try {
|
||||
//如果导入的java对象,需要在这里自己进行变换。
|
||||
if (valType == String.class){
|
||||
String s = String.valueOf(val.toString());
|
||||
if(StringUtils.endsWith(s, ".0")){
|
||||
val = StringUtils.substringBefore(s, ".0");
|
||||
}else{
|
||||
val = String.valueOf(val.toString());
|
||||
}
|
||||
}else if (valType == Integer.class){
|
||||
val = Double.valueOf(val.toString()).intValue();
|
||||
}else if (valType == Long.class){
|
||||
val = Double.valueOf(val.toString()).longValue();
|
||||
}else if (valType == Double.class){
|
||||
val = Double.valueOf(val.toString());
|
||||
}else if (valType == Float.class){
|
||||
val = Float.valueOf(val.toString());
|
||||
}else if (valType == Date.class){
|
||||
SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
|
||||
val=sdf.parse(val.toString());
|
||||
}else{
|
||||
if (ef.fieldType() != Class.class){
|
||||
val = ef.fieldType().getMethod("getValue", String.class).invoke(null, val.toString());
|
||||
}else{
|
||||
val = Class.forName(this.getClass().getName().replaceAll(this.getClass().getSimpleName(),
|
||||
"fieldtype."+valType.getSimpleName()+"Type")).getMethod("getValue", String.class).invoke(null, val.toString());
|
||||
}
|
||||
}
|
||||
} catch (Exception ex) {
|
||||
log.info("Get cell value ["+i+","+column+"] error: " + ex.toString());
|
||||
val = null;
|
||||
}
|
||||
// set entity value
|
||||
if (os[1] instanceof Field){
|
||||
Reflections.invokeSetter(e, ((Field)os[1]).getName(), val);
|
||||
}else if (os[1] instanceof Method){
|
||||
String mthodName = ((Method)os[1]).getName();
|
||||
if ("get".equals(mthodName.substring(0, 3))){
|
||||
mthodName = "set"+StringUtils.substringAfter(mthodName, "get");
|
||||
}
|
||||
Reflections.invokeMethod(e, mthodName, new Class[] {valType}, new Object[] {val});
|
||||
}
|
||||
}
|
||||
sb.append(val+", ");
|
||||
}
|
||||
dataList.add(e);
|
||||
log.debug("Read success: ["+i+"] "+sb.toString());
|
||||
}
|
||||
return dataList;
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,59 @@
|
|||
/**
|
||||
* Copyright © 2015-2020 <a href="http://www.jeeplus.org/">JeePlus</a> All rights reserved.
|
||||
*/
|
||||
package com.guwan.backend.core.utils.excel.annotation;
|
||||
|
||||
import java.lang.annotation.ElementType;
|
||||
import java.lang.annotation.Retention;
|
||||
import java.lang.annotation.RetentionPolicy;
|
||||
import java.lang.annotation.Target;
|
||||
|
||||
/**
|
||||
* Excel注解定义
|
||||
* @author jeeplus
|
||||
* @version 2016-03-10
|
||||
*/
|
||||
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.TYPE})
|
||||
@Retention(RetentionPolicy.RUNTIME)
|
||||
public @interface ExcelField {
|
||||
|
||||
/**
|
||||
* 导出字段名(默认调用当前字段的“get”方法,如指定导出字段为对象,请填写“对象名.对象属性”,例:“area.name”、“office.name”)
|
||||
*/
|
||||
String value() default "";
|
||||
|
||||
/**
|
||||
* 导出字段标题(需要添加批注请用“**”分隔,标题**批注,仅对导出模板有效)
|
||||
*/
|
||||
String title();
|
||||
|
||||
/**
|
||||
* 字段类型(0:导出导入;1:仅导出;2:仅导入)
|
||||
*/
|
||||
int type() default 0;
|
||||
|
||||
/**
|
||||
* 导出字段对齐方式(0:自动;1:靠左;2:居中;3:靠右)
|
||||
*/
|
||||
int align() default 0;
|
||||
|
||||
/**
|
||||
* 导出字段字段排序(升序)
|
||||
*/
|
||||
int sort() default 0;
|
||||
|
||||
/**
|
||||
* 如果是字典类型,请设置字典的type值
|
||||
*/
|
||||
String dictType() default "";
|
||||
|
||||
/**
|
||||
* 反射类型
|
||||
*/
|
||||
Class<?> fieldType() default Class.class;
|
||||
|
||||
/**
|
||||
* 字段归属组(根据分组导出导入)
|
||||
*/
|
||||
int[] groups() default {};
|
||||
}
|
|
@ -0,0 +1,57 @@
|
|||
/**
|
||||
* Copyright © 2015-2020 <a href="http://www.jeeplus.org/">JeePlus</a> All rights reserved.
|
||||
*/
|
||||
package com.guwan.backend.core.utils.excel.fieldtype;
|
||||
|
||||
import com.google.common.collect.Lists;
|
||||
import com.guwan.backend.core.utils.StringUtils;
|
||||
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* 字段类型转换
|
||||
* @author jeeplus
|
||||
* @version 2016-5-29
|
||||
*/
|
||||
public class ListType {
|
||||
|
||||
/**
|
||||
* 获取对象值(导入)
|
||||
*/
|
||||
public static Object getValue(String val) {
|
||||
List<String> list = Lists.newArrayList();
|
||||
if(!StringUtils.isBlank(val)) {
|
||||
for (String s : val.split(",")) {
|
||||
list.add(s);
|
||||
}
|
||||
}
|
||||
return list;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置对象值(导出)
|
||||
*/
|
||||
public static String setValue(Object val) {
|
||||
if (val != null){
|
||||
List<String> list = (List<String>)val;
|
||||
StringBuffer sb = null;
|
||||
for (String item: list){
|
||||
if(StringUtils.isBlank(item)){
|
||||
continue;
|
||||
}
|
||||
if(sb == null){
|
||||
sb = new StringBuffer(item);
|
||||
}else{
|
||||
sb.append(",").append(item);
|
||||
}
|
||||
}
|
||||
|
||||
if(sb!=null) {
|
||||
return sb.toString().replace("[]", "");
|
||||
}
|
||||
}
|
||||
return "";
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,37 @@
|
|||
package com.guwan.backend.core.utils.file;
|
||||
|
||||
import java.security.MessageDigest;
|
||||
|
||||
|
||||
/**
|
||||
* MD5工具类
|
||||
* ClassName: MD5Util <br/>
|
||||
* date: 2018年1月13日 下午6:54:53 <br/>
|
||||
*
|
||||
* @author Bool
|
||||
* @version
|
||||
*/
|
||||
public class Md5Util {
|
||||
|
||||
|
||||
/**
|
||||
* 简单MD5
|
||||
* @param str
|
||||
* @return
|
||||
*/
|
||||
public static String md5(String str) {
|
||||
|
||||
try {
|
||||
MessageDigest md = MessageDigest.getInstance("MD5");
|
||||
byte[] array = md.digest(str.getBytes("UTF-8"));
|
||||
StringBuilder sb = new StringBuilder();
|
||||
for (byte item : array) {
|
||||
sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));
|
||||
}
|
||||
return sb.toString();
|
||||
}catch(Exception e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,58 @@
|
|||
package com.guwan.backend.core.utils.passwd;
|
||||
|
||||
|
||||
|
||||
import com.guwan.backend.core.utils.file.Md5Util;
|
||||
import org.apache.commons.lang3.RandomStringUtils;
|
||||
|
||||
/**
|
||||
* 通用的密码处理类,用于生成密码和校验密码
|
||||
* ClassName: PassGenerator <br/>
|
||||
* date: 2017年12月13日 下午7:13:03 <br/>
|
||||
*
|
||||
* @author Bool
|
||||
* @version
|
||||
*/
|
||||
public class PassHandler {
|
||||
|
||||
/**
|
||||
* checkPass:校验密码是否一致
|
||||
* @author Bool
|
||||
* @param inputPass 用户传入的密码
|
||||
* @param salt 数据库保存的密码随机码
|
||||
* @param pass 数据库保存的密码MD5
|
||||
* @return
|
||||
*/
|
||||
public static boolean checkPass(String inputPass , String salt , String pass){
|
||||
String pwdMd5 = Md5Util.md5(inputPass);
|
||||
return Md5Util.md5(pwdMd5 + salt).equals(pass);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* buildPassword:用于用户注册时产生一个密码
|
||||
* @author Bool
|
||||
* @param inputPass 输入的密码
|
||||
* @return PassInfo 返回一个密码对象,记得保存
|
||||
*/
|
||||
public static PassInfo buildPassword(String inputPass) {
|
||||
|
||||
//产生一个6位数的随机码
|
||||
String salt = RandomStringUtils.randomAlphabetic(6);
|
||||
//加密后的密码
|
||||
String encryptPassword = Md5Util.md5(Md5Util.md5(inputPass)+salt);
|
||||
//返回对象
|
||||
return new PassInfo(salt,encryptPassword);
|
||||
}
|
||||
|
||||
|
||||
public static void main(String[] args) {
|
||||
|
||||
PassInfo info = buildPassword("190601");
|
||||
|
||||
System.out.println(info.getPassword());
|
||||
System.out.println(info.getSalt());
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,38 @@
|
|||
package com.guwan.backend.core.utils.passwd;
|
||||
|
||||
/**
|
||||
* 密码实体
|
||||
* ClassName: PassInfo <br/>
|
||||
* date: 2018年2月13日 下午7:13:50 <br/>
|
||||
*
|
||||
* @author Bool
|
||||
* @version
|
||||
*/
|
||||
public class PassInfo {
|
||||
|
||||
//密码随机串码
|
||||
private String salt;
|
||||
|
||||
//MD5后的密码
|
||||
private String password;
|
||||
|
||||
public PassInfo(String salt, String password) {
|
||||
super();
|
||||
this.salt = salt;
|
||||
this.password = password;
|
||||
}
|
||||
|
||||
public String getSalt() {
|
||||
return salt;
|
||||
}
|
||||
public void setSalt(String salt) {
|
||||
this.salt = salt;
|
||||
}
|
||||
public String getPassword() {
|
||||
return password;
|
||||
}
|
||||
public void setPassword(String password) {
|
||||
this.password = password;
|
||||
}
|
||||
}
|
||||
|
|
@ -1,7 +1,8 @@
|
|||
package com.guwan.backend.face.entity;
|
||||
|
||||
|
||||
import com.guwan.face.util.UserInfo;
|
||||
|
||||
import com.guwan.backend.face.util.UserInfo;
|
||||
import lombok.Data;
|
||||
|
||||
|
||||
|
|
|
@ -1,7 +1,8 @@
|
|||
package com.guwan.backend.face.enums;
|
||||
|
||||
|
||||
import com.guwan.face.rpc.ErrorCode;
|
||||
|
||||
import com.guwan.backend.face.rpc.ErrorCode;
|
||||
import lombok.Getter;
|
||||
|
||||
|
||||
|
|
|
@ -6,8 +6,9 @@ import com.arcsoft.face.enums.ErrorInfo;
|
|||
import com.arcsoft.face.enums.ExtractType;
|
||||
import com.arcsoft.face.toolkit.ImageFactory;
|
||||
import com.arcsoft.face.toolkit.ImageInfo;
|
||||
import com.guwan.face.config.ArcFaceAutoConfiguration;
|
||||
import com.guwan.face.factory.FaceEngineFactory;
|
||||
|
||||
import com.guwan.backend.face.config.ArcFaceAutoConfiguration;
|
||||
import com.guwan.backend.face.factory.FaceEngineFactory;
|
||||
import lombok.Data;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
|
|
|
@ -3,9 +3,10 @@ package com.guwan.backend.face.factory;
|
|||
import com.arcsoft.face.EngineConfiguration;
|
||||
import com.arcsoft.face.FaceEngine;
|
||||
import com.arcsoft.face.enums.ErrorInfo;
|
||||
import com.guwan.face.config.ArcFaceAutoConfiguration;
|
||||
import com.guwan.face.enums.ErrorCodeEnum;
|
||||
import com.guwan.face.rpc.BusinessException;
|
||||
|
||||
import com.guwan.backend.face.config.ArcFaceAutoConfiguration;
|
||||
import com.guwan.backend.face.enums.ErrorCodeEnum;
|
||||
import com.guwan.backend.face.rpc.BusinessException;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.apache.commons.pool2.BasePooledObjectFactory;
|
||||
|
|
|
@ -1,7 +1,9 @@
|
|||
/*
|
||||
package com.guwan.backend.face.rpc;
|
||||
|
||||
|
||||
import com.guwan.face.enums.ErrorCodeEnum;
|
||||
|
||||
import com.guwan.backend.face.enums.ErrorCodeEnum;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.springframework.web.bind.annotation.ExceptionHandler;
|
||||
import org.springframework.web.bind.annotation.RestControllerAdvice;
|
||||
|
@ -11,9 +13,11 @@ import org.springframework.web.bind.annotation.RestControllerAdvice;
|
|||
public class GlobalExceptionHandler{
|
||||
|
||||
|
||||
/**
|
||||
*/
|
||||
/**
|
||||
* 自定义异常
|
||||
*/
|
||||
*//*
|
||||
|
||||
@ExceptionHandler(BusinessException.class)
|
||||
public Response businessException(BusinessException e) {
|
||||
log.error(e.getMessage(), e);
|
||||
|
@ -45,3 +49,4 @@ public class GlobalExceptionHandler{
|
|||
|
||||
|
||||
}
|
||||
*/
|
||||
|
|
|
@ -4,9 +4,10 @@ package com.guwan.backend.face.service;
|
|||
import com.arcsoft.face.FaceInfo;
|
||||
import com.arcsoft.face.enums.ExtractType;
|
||||
import com.arcsoft.face.toolkit.ImageInfo;
|
||||
import com.guwan.face.entity.ProcessInfo;
|
||||
import com.guwan.face.entity.UserCompareInfo;
|
||||
import com.guwan.face.util.UserInfo;
|
||||
import com.guwan.backend.face.entity.ProcessInfo;
|
||||
import com.guwan.backend.face.entity.UserCompareInfo;
|
||||
import com.guwan.backend.face.util.UserInfo;
|
||||
|
||||
|
||||
import java.util.List;
|
||||
|
||||
|
|
|
@ -7,13 +7,14 @@ import com.arcsoft.face.enums.DetectOrient;
|
|||
import com.arcsoft.face.enums.ExtractType;
|
||||
import com.arcsoft.face.toolkit.ImageInfo;
|
||||
import com.google.common.collect.Lists;
|
||||
import com.guwan.face.entity.ProcessInfo;
|
||||
import com.guwan.face.entity.UserCompareInfo;
|
||||
import com.guwan.face.enums.ErrorCodeEnum;
|
||||
import com.guwan.face.factory.FaceEngineFactory;
|
||||
import com.guwan.face.rpc.BusinessException;
|
||||
import com.guwan.face.service.FaceEngineService;
|
||||
import com.guwan.face.util.UserInfo;
|
||||
|
||||
import com.guwan.backend.face.entity.ProcessInfo;
|
||||
import com.guwan.backend.face.entity.UserCompareInfo;
|
||||
import com.guwan.backend.face.enums.ErrorCodeEnum;
|
||||
import com.guwan.backend.face.factory.FaceEngineFactory;
|
||||
import com.guwan.backend.face.rpc.BusinessException;
|
||||
import com.guwan.backend.face.service.FaceEngineService;
|
||||
import com.guwan.backend.face.util.UserInfo;
|
||||
import jakarta.annotation.PostConstruct;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.apache.commons.pool2.impl.GenericObjectPool;
|
||||
|
|
|
@ -2,6 +2,7 @@ package com.guwan.backend.mapper;
|
|||
|
||||
import com.guwan.backend.pojo.dto.BSCategory;
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import org.apache.ibatis.annotations.Mapper;
|
||||
|
||||
/**
|
||||
* @author 12455
|
||||
|
@ -9,6 +10,7 @@ import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
|||
* @createDate 2025-03-13 23:00:51
|
||||
* @Entity com.guwan.backend.pojo.dto.BSCategory
|
||||
*/
|
||||
@Mapper
|
||||
public interface BSCategoryMapper extends BaseMapper<BSCategory> {
|
||||
|
||||
}
|
||||
|
|
|
@ -2,6 +2,7 @@ package com.guwan.backend.mapper;
|
|||
|
||||
import com.guwan.backend.pojo.Courses;
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import org.apache.ibatis.annotations.Mapper;
|
||||
|
||||
/**
|
||||
* @author 12455
|
||||
|
@ -9,6 +10,7 @@ import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
|||
* @createDate 2025-03-13 22:45:19
|
||||
* @Entity com.guwan.backend.pojo.Courses
|
||||
*/
|
||||
@Mapper
|
||||
public interface CoursesMapper extends BaseMapper<Courses> {
|
||||
|
||||
}
|
||||
|
|
|
@ -2,6 +2,7 @@ package com.guwan.backend.mapper;
|
|||
|
||||
import com.guwan.backend.pojo.entity.Papers;
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import org.apache.ibatis.annotations.Mapper;
|
||||
|
||||
/**
|
||||
* @author 12455
|
||||
|
@ -9,6 +10,7 @@ import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
|||
* @createDate 2025-01-11 12:00:45
|
||||
* @Entity com.guwan.backend.pojo.entity.Papers
|
||||
*/
|
||||
@Mapper
|
||||
public interface PapersMapper extends BaseMapper<Papers> {
|
||||
|
||||
}
|
||||
|
|
|
@ -2,6 +2,7 @@ package com.guwan.backend.mapper;
|
|||
|
||||
import com.guwan.backend.pojo.entity.Questions;
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import org.apache.ibatis.annotations.Mapper;
|
||||
|
||||
/**
|
||||
* @author 12455
|
||||
|
@ -9,6 +10,7 @@ import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
|||
* @createDate 2025-01-11 17:53:58
|
||||
* @Entity com.guwan.backend.pojo.entity.Questions
|
||||
*/
|
||||
@Mapper
|
||||
public interface QuestionsMapper extends BaseMapper<Questions> {
|
||||
|
||||
}
|
||||
|
|
|
@ -0,0 +1,70 @@
|
|||
package com.guwan.backend.model.exam.controller;
|
||||
|
||||
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
|
||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||
import com.guwan.backend.core.api.ApiRest;
|
||||
import com.guwan.backend.core.api.controller.BaseController;
|
||||
import com.guwan.backend.core.api.dto.BaseIdReqDTO;
|
||||
import com.guwan.backend.core.api.dto.BaseIdsReqDTO;
|
||||
import com.guwan.backend.core.api.dto.BaseStateReqDTO;
|
||||
import com.guwan.backend.core.api.dto.PagingReqDTO;
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import com.guwan.backend.model.exam.dto.request.ExamSaveReqDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamOnlineRespDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamReviewRespDTO;
|
||||
import com.guwan.backend.model.exam.entity.Exam;
|
||||
import com.guwan.backend.model.exam.service.ExamService;
|
||||
|
||||
|
||||
import io.swagger.annotations.Api;
|
||||
import io.swagger.annotations.ApiOperation;
|
||||
//import org.apache.shiro.authz.annotation.RequiresRoles;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.web.bind.annotation.RequestBody;
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.RequestMethod;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试控制器
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Api(tags={"考试"})
|
||||
@RestController
|
||||
@RequestMapping("/api/common/exam/api/exam/exam")
|
||||
public class ExamController extends BaseController {
|
||||
|
||||
@Autowired
|
||||
private ExamService baseService;
|
||||
|
||||
|
||||
/**
|
||||
* 分页查找
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "考试视角")
|
||||
@RequestMapping(value = "/online-paging", method = { RequestMethod.POST})
|
||||
public ApiRest<IPage<ExamOnlineRespDTO>> myPaging(@RequestBody PagingReqDTO<ExamDTO> reqDTO) {
|
||||
|
||||
//分页查询并转换
|
||||
IPage<ExamOnlineRespDTO> page = baseService.onlinePaging(reqDTO);
|
||||
return super.success(page);
|
||||
}
|
||||
|
||||
|
||||
@ApiOperation(value = "查找详情")
|
||||
@RequestMapping(value = "/detail", method = { RequestMethod.POST})
|
||||
public ApiRest<ExamSaveReqDTO> find(@RequestBody BaseIdReqDTO reqDTO) {
|
||||
ExamSaveReqDTO dto = baseService.findDetail(reqDTO.getId());
|
||||
return super.success(dto);
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,103 @@
|
|||
package com.guwan.backend.model.exam.dto;
|
||||
|
||||
import com.fasterxml.jackson.annotation.JsonFormat;
|
||||
|
||||
|
||||
import com.guwan.backend.model.exam.enums.ExamState;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
import org.springframework.format.annotation.DateTimeFormat;
|
||||
|
||||
import java.io.Serializable;
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试数据传输类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="考试", description="考试")
|
||||
public class ExamDTO implements Serializable {
|
||||
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "ID", required=true)
|
||||
private String id;
|
||||
|
||||
@ApiModelProperty(value = "考试名称", required=true)
|
||||
private String title;
|
||||
|
||||
@ApiModelProperty(value = "考试描述", required=true)
|
||||
private String content;
|
||||
|
||||
@ApiModelProperty(value = "1公开2部门3定员", required=true)
|
||||
private Integer openType;
|
||||
|
||||
@ApiModelProperty(value = "考试状态", required=true)
|
||||
private Integer state;
|
||||
|
||||
@ApiModelProperty(value = "是否限时", required=true)
|
||||
private Boolean timeLimit;
|
||||
|
||||
@JsonFormat(timezone = "GMT+8", pattern = "yyyy-MM-dd")
|
||||
@DateTimeFormat(pattern = "yyyy-MM-dd")
|
||||
@ApiModelProperty(value = "开始时间", required=true)
|
||||
private Date startTime;
|
||||
|
||||
@JsonFormat(timezone = "GMT+8", pattern = "yyyy-MM-dd")
|
||||
@DateTimeFormat(pattern = "yyyy-MM-dd")
|
||||
@ApiModelProperty(value = "结束时间", required=true)
|
||||
private Date endTime;
|
||||
|
||||
@ApiModelProperty(value = "创建时间", required=true)
|
||||
private Date createTime;
|
||||
|
||||
@ApiModelProperty(value = "更新时间", required=true)
|
||||
private Date updateTime;
|
||||
|
||||
@ApiModelProperty(value = "总分数", required=true)
|
||||
private Integer totalScore;
|
||||
|
||||
@ApiModelProperty(value = "总时长(分钟)", required=true)
|
||||
private Integer totalTime;
|
||||
|
||||
@ApiModelProperty(value = "及格分数", required=true)
|
||||
private Integer qualifyScore;
|
||||
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 是否结束
|
||||
* @return
|
||||
*/
|
||||
public Integer getState(){
|
||||
|
||||
if(this.timeLimit!=null && this.timeLimit){
|
||||
|
||||
if(System.currentTimeMillis() < startTime.getTime() ){
|
||||
return ExamState.READY_START;
|
||||
}
|
||||
|
||||
if(System.currentTimeMillis() > endTime.getTime()){
|
||||
return ExamState.OVERDUE;
|
||||
}
|
||||
|
||||
if(System.currentTimeMillis() > startTime.getTime()
|
||||
&& System.currentTimeMillis() < endTime.getTime()
|
||||
&& !ExamState.DISABLED.equals(this.state)){
|
||||
return ExamState.ENABLE;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
return this.state;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.exam.dto;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试部门数据传输类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-03 17:24
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="考试部门", description="考试部门")
|
||||
public class ExamDepartDTO implements Serializable {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "ID", required=true)
|
||||
private String id;
|
||||
|
||||
@ApiModelProperty(value = "考试ID", required=true)
|
||||
private String examId;
|
||||
|
||||
@ApiModelProperty(value = "部门ID", required=true)
|
||||
private String departId;
|
||||
|
||||
}
|
|
@ -0,0 +1,51 @@
|
|||
package com.guwan.backend.model.exam.dto;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试题库数据传输类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-05 11:14
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="考试题库", description="考试题库")
|
||||
public class ExamRepoDTO implements Serializable {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "ID", required=true)
|
||||
private String id;
|
||||
|
||||
@ApiModelProperty(value = "考试ID", required=true)
|
||||
private String examId;
|
||||
|
||||
@ApiModelProperty(value = "题库ID", required=true)
|
||||
private String repoId;
|
||||
|
||||
@ApiModelProperty(value = "单选题数量", required=true)
|
||||
private Integer radioCount;
|
||||
|
||||
@ApiModelProperty(value = "单选题分数", required=true)
|
||||
private Integer radioScore;
|
||||
|
||||
@ApiModelProperty(value = "多选题数量", required=true)
|
||||
private Integer multiCount;
|
||||
|
||||
@ApiModelProperty(value = "多选题分数", required=true)
|
||||
private Integer multiScore;
|
||||
|
||||
@ApiModelProperty(value = "判断题数量", required=true)
|
||||
private Integer judgeCount;
|
||||
|
||||
@ApiModelProperty(value = "判断题分数", required=true)
|
||||
private Integer judgeScore;
|
||||
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.exam.dto.ext;
|
||||
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ExamRepoDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试题库数据传输类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-05 11:14
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="考试题库扩展响应类", description="考试题库扩展响应类")
|
||||
public class ExamRepoExtDTO extends ExamRepoDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "单选题总量", required=true)
|
||||
private Integer totalRadio;
|
||||
|
||||
@ApiModelProperty(value = "多选题总量", required=true)
|
||||
private Integer totalMulti;
|
||||
|
||||
@ApiModelProperty(value = "判断题总量", required=true)
|
||||
private Integer totalJudge;
|
||||
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.exam.dto.request;
|
||||
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import com.guwan.backend.model.exam.dto.ext.ExamRepoExtDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试保存请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="考试保存请求类", description="考试保存请求类")
|
||||
public class ExamSaveReqDTO extends ExamDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "题库列表", required=true)
|
||||
private List<ExamRepoExtDTO> repoList;
|
||||
|
||||
@ApiModelProperty(value = "考试部门列表", required=true)
|
||||
private List<String> departIds;
|
||||
|
||||
}
|
|
@ -0,0 +1,23 @@
|
|||
package com.guwan.backend.model.exam.dto.response;
|
||||
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试分页响应类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="在线考试分页响应类", description="在线考试分页响应类")
|
||||
public class ExamOnlineRespDTO extends ExamDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,32 @@
|
|||
package com.guwan.backend.model.exam.dto.response;
|
||||
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试分页响应类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="阅卷分页响应类", description="阅卷分页响应类")
|
||||
public class ExamReviewRespDTO extends ExamDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "考试人数", required=true)
|
||||
private Integer examUser;
|
||||
|
||||
@ApiModelProperty(value = "待阅试卷", required=true)
|
||||
private Integer unreadPaper;
|
||||
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,101 @@
|
|||
package com.guwan.backend.model.exam.entity;
|
||||
|
||||
import com.baomidou.mybatisplus.annotation.IdType;
|
||||
import com.baomidou.mybatisplus.annotation.TableField;
|
||||
import com.baomidou.mybatisplus.annotation.TableId;
|
||||
import com.baomidou.mybatisplus.annotation.TableName;
|
||||
import com.baomidou.mybatisplus.extension.activerecord.Model;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试实体类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Data
|
||||
@TableName("el_exam")
|
||||
public class Exam extends Model<Exam> {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/**
|
||||
* ID
|
||||
*/
|
||||
@TableId(value = "id", type = IdType.ASSIGN_ID)
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* 考试名称
|
||||
*/
|
||||
private String title;
|
||||
|
||||
/**
|
||||
* 考试描述
|
||||
*/
|
||||
private String content;
|
||||
|
||||
/**
|
||||
* 1公开2部门3定员
|
||||
*/
|
||||
@TableField("open_type")
|
||||
private Integer openType;
|
||||
|
||||
/**
|
||||
* 考试状态
|
||||
*/
|
||||
private Integer state;
|
||||
|
||||
/**
|
||||
* 是否限时
|
||||
*/
|
||||
@TableField("time_limit")
|
||||
private Boolean timeLimit;
|
||||
|
||||
/**
|
||||
* 开始时间
|
||||
*/
|
||||
@TableField("start_time")
|
||||
private Date startTime;
|
||||
|
||||
/**
|
||||
* 结束时间
|
||||
*/
|
||||
@TableField("end_time")
|
||||
private Date endTime;
|
||||
|
||||
/**
|
||||
* 创建时间
|
||||
*/
|
||||
@TableField("create_time")
|
||||
private Date createTime;
|
||||
|
||||
/**
|
||||
* 更新时间
|
||||
*/
|
||||
@TableField("update_time")
|
||||
private Date updateTime;
|
||||
|
||||
/**
|
||||
* 总分数
|
||||
*/
|
||||
@TableField("total_score")
|
||||
private Integer totalScore;
|
||||
|
||||
/**
|
||||
* 总时长(分钟)
|
||||
*/
|
||||
@TableField("total_time")
|
||||
private Integer totalTime;
|
||||
|
||||
/**
|
||||
* 及格分数
|
||||
*/
|
||||
@TableField("qualify_score")
|
||||
private Integer qualifyScore;
|
||||
|
||||
}
|
|
@ -0,0 +1,50 @@
|
|||
package com.guwan.backend.model.exam.entity;
|
||||
|
||||
import com.baomidou.mybatisplus.annotation.IdType;
|
||||
import com.baomidou.mybatisplus.annotation.TableField;
|
||||
import com.baomidou.mybatisplus.annotation.TableId;
|
||||
import com.baomidou.mybatisplus.annotation.TableName;
|
||||
import com.baomidou.mybatisplus.extension.activerecord.Model;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试部门实体类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-03 17:24
|
||||
*/
|
||||
@Data
|
||||
@TableName("el_exam_depart")
|
||||
/**
|
||||
* 当你的实体类 继承 Model<T> 后,就可以直接使用 MyBatis-Plus 提供的 Active Record (AR) 方法,例如:
|
||||
*
|
||||
* insert() —— 直接插入数据
|
||||
* deleteById() —— 按 ID 删除
|
||||
* selectById() —— 按 ID 查询
|
||||
* updateById() —— 按 ID 更新
|
||||
*/
|
||||
public class ExamDepart extends Model<ExamDepart> {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/**
|
||||
* ID
|
||||
*/
|
||||
@TableId(value = "id", type = IdType.ASSIGN_ID)
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* 考试ID
|
||||
*/
|
||||
@TableField("exam_id")
|
||||
private String examId;
|
||||
|
||||
/**
|
||||
* 部门ID
|
||||
*/
|
||||
@TableField("depart_id")
|
||||
private String departId;
|
||||
|
||||
}
|
|
@ -0,0 +1,78 @@
|
|||
package com.guwan.backend.model.exam.entity;
|
||||
|
||||
import com.baomidou.mybatisplus.annotation.IdType;
|
||||
import com.baomidou.mybatisplus.annotation.TableField;
|
||||
import com.baomidou.mybatisplus.annotation.TableId;
|
||||
import com.baomidou.mybatisplus.annotation.TableName;
|
||||
import com.baomidou.mybatisplus.extension.activerecord.Model;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试题库实体类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-05 11:14
|
||||
*/
|
||||
@Data
|
||||
@TableName("el_exam_repo")
|
||||
public class ExamRepo extends Model<ExamRepo> {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/**
|
||||
* ID
|
||||
*/
|
||||
@TableId(value = "id", type = IdType.ASSIGN_ID)
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* 考试ID
|
||||
*/
|
||||
@TableField("exam_id")
|
||||
private String examId;
|
||||
|
||||
/**
|
||||
* 题库ID
|
||||
*/
|
||||
@TableField("repo_id")
|
||||
private String repoId;
|
||||
|
||||
/**
|
||||
* 单选题数量
|
||||
*/
|
||||
@TableField("radio_count")
|
||||
private Integer radioCount;
|
||||
|
||||
/**
|
||||
* 单选题分数
|
||||
*/
|
||||
@TableField("radio_score")
|
||||
private Integer radioScore;
|
||||
|
||||
/**
|
||||
* 多选题数量
|
||||
*/
|
||||
@TableField("multi_count")
|
||||
private Integer multiCount;
|
||||
|
||||
/**
|
||||
* 多选题分数
|
||||
*/
|
||||
@TableField("multi_score")
|
||||
private Integer multiScore;
|
||||
|
||||
/**
|
||||
* 判断题数量
|
||||
*/
|
||||
@TableField("judge_count")
|
||||
private Integer judgeCount;
|
||||
|
||||
/**
|
||||
* 判断题分数
|
||||
*/
|
||||
@TableField("judge_score")
|
||||
private Integer judgeScore;
|
||||
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.exam.enums;
|
||||
|
||||
|
||||
/**
|
||||
* 考试状态
|
||||
* @author bool
|
||||
* @date 2019-10-30 13:11
|
||||
*/
|
||||
public interface ExamState {
|
||||
|
||||
|
||||
/**
|
||||
* 考试中
|
||||
*/
|
||||
Integer ENABLE = 0;
|
||||
|
||||
/**
|
||||
* 待阅卷
|
||||
*/
|
||||
Integer DISABLED = 1;
|
||||
|
||||
/**
|
||||
* 已完成
|
||||
*/
|
||||
Integer READY_START = 2;
|
||||
|
||||
/**
|
||||
* 已结束
|
||||
*/
|
||||
Integer OVERDUE = 3;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,16 @@
|
|||
package com.guwan.backend.model.exam.mapper;
|
||||
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import com.guwan.backend.model.exam.entity.ExamDepart;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试部门Mapper
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-03 17:24
|
||||
*/
|
||||
public interface ExamDepartMapper extends BaseMapper<ExamDepart> {
|
||||
|
||||
}
|
|
@ -0,0 +1,32 @@
|
|||
package com.guwan.backend.model.exam.mapper;
|
||||
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamOnlineRespDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamReviewRespDTO;
|
||||
import com.guwan.backend.model.exam.entity.Exam;
|
||||
import org.apache.ibatis.annotations.Param;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试Mapper
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
public interface ExamMapper extends BaseMapper<Exam> {
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 在线考试分页响应类-考生视角
|
||||
* @param page
|
||||
* @param query
|
||||
* @return
|
||||
*/
|
||||
IPage<ExamOnlineRespDTO> online(Page page, @Param("query") ExamDTO query);
|
||||
}
|
|
@ -0,0 +1,27 @@
|
|||
package com.guwan.backend.model.exam.mapper;
|
||||
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ext.ExamRepoExtDTO;
|
||||
import com.guwan.backend.model.exam.entity.ExamRepo;
|
||||
import org.apache.ibatis.annotations.Param;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试题库Mapper
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-05 11:14
|
||||
*/
|
||||
public interface ExamRepoMapper extends BaseMapper<ExamRepo> {
|
||||
|
||||
/**
|
||||
* 查找考试题库列表
|
||||
* @param examId
|
||||
* @return
|
||||
*/
|
||||
List<ExamRepoExtDTO> listByExam(@Param("examId") String examId);
|
||||
}
|
|
@ -0,0 +1,26 @@
|
|||
package com.guwan.backend.model.exam.service;
|
||||
|
||||
import com.baomidou.mybatisplus.extension.service.IService;
|
||||
import com.guwan.backend.model.exam.entity.ExamDepart;
|
||||
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试部门业务类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-03 17:24
|
||||
*/
|
||||
public interface ExamDepartService extends IService<ExamDepart> {
|
||||
|
||||
|
||||
/**
|
||||
* 根据考试查找对应的部门
|
||||
* @param examId
|
||||
* @return
|
||||
*/
|
||||
List<String> listByExam(String examId);
|
||||
}
|
|
@ -0,0 +1,30 @@
|
|||
package com.guwan.backend.model.exam.service;
|
||||
|
||||
import com.baomidou.mybatisplus.extension.service.IService;
|
||||
import com.guwan.backend.model.exam.dto.ext.ExamRepoExtDTO;
|
||||
import com.guwan.backend.model.exam.entity.ExamRepo;
|
||||
|
||||
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试题库业务类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-05 11:14
|
||||
*/
|
||||
public interface ExamRepoService extends IService<ExamRepo> {
|
||||
|
||||
/**
|
||||
* 查找考试题库列表
|
||||
* @param examId
|
||||
* @return
|
||||
*/
|
||||
List<ExamRepoExtDTO> listByExam(String examId);
|
||||
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,34 @@
|
|||
package com.guwan.backend.model.exam.service;
|
||||
|
||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||
import com.baomidou.mybatisplus.extension.service.IService;
|
||||
import com.guwan.backend.core.api.dto.PagingReqDTO;
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import com.guwan.backend.model.exam.dto.request.ExamSaveReqDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamOnlineRespDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamReviewRespDTO;
|
||||
import com.guwan.backend.model.exam.entity.Exam;
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试业务类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
public interface ExamService extends IService<Exam> {
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 在线考试分页响应类-考生视角
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
IPage<ExamOnlineRespDTO> onlinePaging(PagingReqDTO<ExamDTO> reqDTO);
|
||||
|
||||
ExamSaveReqDTO findDetail(String id);
|
||||
|
||||
}
|
|
@ -0,0 +1,45 @@
|
|||
package com.guwan.backend.model.exam.service.impl;
|
||||
|
||||
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
|
||||
|
||||
|
||||
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
||||
import com.guwan.backend.model.exam.entity.ExamDepart;
|
||||
import com.guwan.backend.model.exam.mapper.ExamDepartMapper;
|
||||
import com.guwan.backend.model.exam.service.ExamDepartService;
|
||||
import org.springframework.stereotype.Service;
|
||||
import org.springframework.util.CollectionUtils;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试部门业务实现类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-03 17:24
|
||||
*/
|
||||
@Service
|
||||
public class ExamDepartServiceImpl extends ServiceImpl<ExamDepartMapper, ExamDepart> implements ExamDepartService {
|
||||
|
||||
|
||||
|
||||
@Override
|
||||
public List<String> listByExam(String examId) {
|
||||
// 先删除
|
||||
QueryWrapper<ExamDepart> wrapper = new QueryWrapper<>();
|
||||
wrapper.lambda().eq(ExamDepart::getExamId, examId);
|
||||
List<ExamDepart> list = this.list(wrapper);
|
||||
List<String> ids = new ArrayList<>();
|
||||
if(!CollectionUtils.isEmpty(list)){
|
||||
for(ExamDepart item: list){
|
||||
ids.add(item.getDepartId());
|
||||
}
|
||||
}
|
||||
|
||||
return ids;
|
||||
|
||||
}
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
package com.guwan.backend.model.exam.service.impl;
|
||||
|
||||
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
|
||||
import com.baomidou.mybatisplus.core.toolkit.IdWorker;
|
||||
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
||||
|
||||
import com.guwan.backend.model.exam.dto.ext.ExamRepoExtDTO;
|
||||
import com.guwan.backend.model.exam.entity.ExamRepo;
|
||||
import com.guwan.backend.model.exam.mapper.ExamRepoMapper;
|
||||
import com.guwan.backend.model.exam.service.ExamRepoService;
|
||||
import org.springframework.stereotype.Service;
|
||||
import org.springframework.transaction.annotation.Transactional;
|
||||
import org.springframework.util.CollectionUtils;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试题库业务实现类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-09-05 11:14
|
||||
*/
|
||||
@Service
|
||||
public class ExamRepoServiceImpl extends ServiceImpl<ExamRepoMapper, ExamRepo> implements ExamRepoService {
|
||||
|
||||
|
||||
|
||||
|
||||
@Override
|
||||
public List<ExamRepoExtDTO> listByExam(String examId) {
|
||||
return baseMapper.listByExam(examId);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,77 @@
|
|||
package com.guwan.backend.model.exam.service.impl;
|
||||
|
||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
||||
|
||||
import com.guwan.backend.core.api.dto.PagingReqDTO;
|
||||
|
||||
import com.guwan.backend.core.BeanMapper;
|
||||
import com.guwan.backend.model.exam.dto.ExamDTO;
|
||||
import com.guwan.backend.model.exam.dto.ext.ExamRepoExtDTO;
|
||||
import com.guwan.backend.model.exam.dto.request.ExamSaveReqDTO;
|
||||
import com.guwan.backend.model.exam.dto.response.ExamOnlineRespDTO;
|
||||
import com.guwan.backend.model.exam.entity.Exam;
|
||||
import com.guwan.backend.model.exam.mapper.ExamMapper;
|
||||
import com.guwan.backend.model.exam.service.ExamDepartService;
|
||||
import com.guwan.backend.model.exam.service.ExamRepoService;
|
||||
import com.guwan.backend.model.exam.service.ExamService;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 考试业务实现类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-07-25 16:18
|
||||
*/
|
||||
@Service
|
||||
public class ExamServiceImpl extends ServiceImpl<ExamMapper, Exam> implements ExamService {
|
||||
|
||||
|
||||
@Autowired
|
||||
private ExamRepoService examRepoService;
|
||||
|
||||
@Autowired
|
||||
private ExamDepartService examDepartService;
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
@Override
|
||||
public IPage<ExamOnlineRespDTO> onlinePaging(PagingReqDTO<ExamDTO> reqDTO) {
|
||||
|
||||
|
||||
// 创建分页对象
|
||||
Page page = new Page(reqDTO.getCurrent(), reqDTO.getSize());
|
||||
|
||||
// 查找分页
|
||||
IPage<ExamOnlineRespDTO> pageData = baseMapper.online(page, reqDTO.getParams());
|
||||
|
||||
return pageData;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ExamSaveReqDTO findDetail(String id) {
|
||||
ExamSaveReqDTO respDTO = new ExamSaveReqDTO();
|
||||
Exam exam = this.getById(id);
|
||||
BeanMapper.copy(exam, respDTO);
|
||||
|
||||
// 考试部门
|
||||
List<String> departIds = examDepartService.listByExam(id);
|
||||
respDTO.setDepartIds(departIds);
|
||||
|
||||
// 题库
|
||||
List<ExamRepoExtDTO> repos = examRepoService.listByExam(id);
|
||||
respDTO.setRepoList(repos);
|
||||
|
||||
return respDTO;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,171 @@
|
|||
package com.guwan.backend.model.paper.controller;
|
||||
|
||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||
|
||||
import com.guwan.backend.core.api.ApiRest;
|
||||
import com.guwan.backend.core.api.controller.BaseController;
|
||||
import com.guwan.backend.core.api.dto.BaseIdReqDTO;
|
||||
import com.guwan.backend.core.api.dto.BaseIdRespDTO;
|
||||
import com.guwan.backend.core.api.dto.PagingReqDTO;
|
||||
import com.guwan.backend.model.paper.dto.PaperDTO;
|
||||
import com.guwan.backend.model.paper.dto.ext.PaperQuDetailDTO;
|
||||
import com.guwan.backend.model.paper.dto.request.PaperAnswerDTO;
|
||||
import com.guwan.backend.model.paper.dto.request.PaperCreateReqDTO;
|
||||
import com.guwan.backend.model.paper.dto.request.PaperListReqDTO;
|
||||
import com.guwan.backend.model.paper.dto.request.PaperQuQueryDTO;
|
||||
import com.guwan.backend.model.paper.dto.response.ExamDetailRespDTO;
|
||||
import com.guwan.backend.model.paper.dto.response.ExamResultRespDTO;
|
||||
import com.guwan.backend.model.paper.dto.response.PaperListRespDTO;
|
||||
import com.guwan.backend.model.paper.service.PaperService;
|
||||
import com.guwan.backend.model.user.UserUtils;
|
||||
import com.guwan.backend.security.CustomUserDetails;
|
||||
import com.guwan.backend.util.JwtUtil;
|
||||
import io.swagger.annotations.Api;
|
||||
import io.swagger.annotations.ApiOperation;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.security.core.Authentication;
|
||||
import org.springframework.security.core.context.SecurityContextHolder;
|
||||
import org.springframework.web.bind.annotation.RequestBody;
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.RequestMethod;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷控制器
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 16:33
|
||||
*/
|
||||
@Api(tags={"试卷"})
|
||||
@RestController
|
||||
@RequestMapping("/exam/api/paper/paper")
|
||||
public class PaperController extends BaseController {
|
||||
|
||||
@Autowired
|
||||
private PaperService baseService;
|
||||
|
||||
@Autowired
|
||||
private JwtUtil jwtUtil;
|
||||
|
||||
/**
|
||||
* 分页查找
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "分页查找")
|
||||
@RequestMapping(value = "/paging", method = { RequestMethod.POST})
|
||||
public ApiRest<IPage<PaperListRespDTO>> paging(@RequestBody PagingReqDTO<PaperListReqDTO> reqDTO) {
|
||||
//分页查询并转换
|
||||
IPage<PaperListRespDTO> page = baseService.paging(reqDTO);
|
||||
return super.success(page);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* 创建试卷
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "创建试卷")
|
||||
@RequestMapping(value = "/create-paper", method = { RequestMethod.POST})
|
||||
public ApiRest<BaseIdRespDTO> save(@RequestBody PaperCreateReqDTO reqDTO) {
|
||||
|
||||
|
||||
Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
|
||||
if (authentication != null && authentication.getPrincipal() instanceof CustomUserDetails userDetails) {
|
||||
// sysLog.setUserId(userDetails.getUserId());
|
||||
// sysLog.setUsername(userDetails.getUsername());
|
||||
}
|
||||
|
||||
|
||||
//复制参数
|
||||
// String paperId = baseService.createPaper(UserUtils.getUserId(), reqDTO.getExamId());
|
||||
String paperId = baseService.createPaper(jwtUtil.getUserIdFromToken("111") + "",
|
||||
reqDTO.getExamId());
|
||||
|
||||
return super.success(new BaseIdRespDTO(paperId));
|
||||
}
|
||||
|
||||
/**
|
||||
* 批量删除
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "试卷详情")
|
||||
@RequestMapping(value = "/paper-detail", method = { RequestMethod.POST})
|
||||
public ApiRest<ExamDetailRespDTO> paperDetail(@RequestBody BaseIdReqDTO reqDTO) {
|
||||
//根据ID删除
|
||||
ExamDetailRespDTO respDTO = baseService.paperDetail(reqDTO.getId());
|
||||
return super.success(respDTO);
|
||||
}
|
||||
|
||||
/**
|
||||
* 批量删除
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "试题详情")
|
||||
@RequestMapping(value = "/qu-detail", method = { RequestMethod.POST})
|
||||
public ApiRest<PaperQuDetailDTO> quDetail(@RequestBody PaperQuQueryDTO reqDTO) {
|
||||
//根据ID删除
|
||||
PaperQuDetailDTO respDTO = baseService.findQuDetail(reqDTO.getPaperId(), reqDTO.getQuId());
|
||||
return super.success(respDTO);
|
||||
}
|
||||
|
||||
/**
|
||||
* 填充答案
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "填充答案")
|
||||
@RequestMapping(value = "/fill-answer", method = { RequestMethod.POST})
|
||||
public ApiRest<PaperQuDetailDTO> fillAnswer(@RequestBody PaperAnswerDTO reqDTO) {
|
||||
//根据ID删除
|
||||
baseService.fillAnswer(reqDTO);
|
||||
return super.success();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 交卷操作
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "交卷操作")
|
||||
@RequestMapping(value = "/hand-exam", method = { RequestMethod.POST})
|
||||
public ApiRest<PaperQuDetailDTO> handleExam(@RequestBody BaseIdReqDTO reqDTO) {
|
||||
//根据ID删除
|
||||
baseService.handExam(reqDTO.getId());
|
||||
return super.success();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 批量删除
|
||||
* @param reqDTO
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "试卷详情")
|
||||
@RequestMapping(value = "/paper-result", method = { RequestMethod.POST})
|
||||
public ApiRest<ExamResultRespDTO> paperResult(@RequestBody BaseIdReqDTO reqDTO) {
|
||||
//根据ID删除
|
||||
ExamResultRespDTO respDTO = baseService.paperResult(reqDTO.getId());
|
||||
return super.success(respDTO);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 检测用户有没有中断的考试
|
||||
* @return
|
||||
*/
|
||||
@ApiOperation(value = "检测进行中的考试")
|
||||
@RequestMapping(value = "/check-process", method = { RequestMethod.POST})
|
||||
public ApiRest<PaperDTO> checkProcess() {
|
||||
//复制参数
|
||||
PaperDTO dto = baseService.checkProcess(UserUtils.getUserId());
|
||||
return super.success(dto);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,80 @@
|
|||
package com.guwan.backend.model.paper.dto;
|
||||
|
||||
|
||||
import com.guwan.backend.core.annon.Dict;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷", description="试卷")
|
||||
public class PaperDTO implements Serializable {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "试卷ID", required=true)
|
||||
private String id;
|
||||
|
||||
@Dict(dictTable = "sys_user", dicText = "real_name", dicCode = "id")
|
||||
@ApiModelProperty(value = "用户ID", required=true)
|
||||
private String userId;
|
||||
|
||||
@Dict(dictTable = "sys_depart", dicText = "dept_name", dicCode = "id")
|
||||
@ApiModelProperty(value = "部门ID", required=true)
|
||||
private String departId;
|
||||
|
||||
@ApiModelProperty(value = "规则ID", required=true)
|
||||
private String examId;
|
||||
|
||||
@ApiModelProperty(value = "考试标题", required=true)
|
||||
private String title;
|
||||
|
||||
@ApiModelProperty(value = "考试时长", required=true)
|
||||
private Integer totalTime;
|
||||
|
||||
@ApiModelProperty(value = "用户时长", required=true)
|
||||
private Integer userTime;
|
||||
|
||||
@ApiModelProperty(value = "试卷总分", required=true)
|
||||
private Integer totalScore;
|
||||
|
||||
@ApiModelProperty(value = "及格分", required=true)
|
||||
private Integer qualifyScore;
|
||||
|
||||
@ApiModelProperty(value = "客观分", required=true)
|
||||
private Integer objScore;
|
||||
|
||||
@ApiModelProperty(value = "主观分", required=true)
|
||||
private Integer subjScore;
|
||||
|
||||
@ApiModelProperty(value = "用户得分", required=true)
|
||||
private Integer userScore;
|
||||
|
||||
@ApiModelProperty(value = "是否包含简答题", required=true)
|
||||
private Boolean hasSaq;
|
||||
|
||||
@ApiModelProperty(value = "试卷状态", required=true)
|
||||
private Integer state;
|
||||
|
||||
@ApiModelProperty(value = "创建时间", required=true)
|
||||
private Date createTime;
|
||||
|
||||
@ApiModelProperty(value = "更新时间", required=true)
|
||||
private Date updateTime;
|
||||
|
||||
@ApiModelProperty(value = "截止时间")
|
||||
private Date limitTime;
|
||||
|
||||
}
|
|
@ -0,0 +1,48 @@
|
|||
package com.guwan.backend.model.paper.dto;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题备选答案请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷考题备选答案", description="试卷考题备选答案")
|
||||
public class PaperQuAnswerDTO implements Serializable {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "自增ID", required=true)
|
||||
private String id;
|
||||
|
||||
@ApiModelProperty(value = "试卷ID", required=true)
|
||||
private String paperId;
|
||||
|
||||
@ApiModelProperty(value = "回答项ID", required=true)
|
||||
private String answerId;
|
||||
|
||||
@ApiModelProperty(value = "题目ID", required=true)
|
||||
private String quId;
|
||||
|
||||
@ApiModelProperty(value = "是否正确项", required=true)
|
||||
private Boolean isRight;
|
||||
|
||||
@ApiModelProperty(value = "是否选中", required=true)
|
||||
private Boolean checked;
|
||||
|
||||
@ApiModelProperty(value = "排序", required=true)
|
||||
private Integer sort;
|
||||
|
||||
@ApiModelProperty(value = "选项标签", required=true)
|
||||
private String abc;
|
||||
|
||||
}
|
|
@ -0,0 +1,54 @@
|
|||
package com.guwan.backend.model.paper.dto;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷考题", description="试卷考题")
|
||||
public class PaperQuDTO implements Serializable {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "ID", required=true)
|
||||
private String id;
|
||||
|
||||
@ApiModelProperty(value = "试卷ID", required=true)
|
||||
private String paperId;
|
||||
|
||||
@ApiModelProperty(value = "题目ID", required=true)
|
||||
private String quId;
|
||||
|
||||
@ApiModelProperty(value = "题目类型", required=true)
|
||||
private Integer quType;
|
||||
|
||||
@ApiModelProperty(value = "是否已答", required=true)
|
||||
private Boolean answered;
|
||||
|
||||
@ApiModelProperty(value = "主观答案", required=true)
|
||||
private String answer;
|
||||
|
||||
@ApiModelProperty(value = "问题排序", required=true)
|
||||
private Integer sort;
|
||||
|
||||
@ApiModelProperty(value = "单题分分值", required=true)
|
||||
private Integer score;
|
||||
|
||||
@ApiModelProperty(value = "实际得分(主观题)", required=true)
|
||||
private Integer actualScore;
|
||||
|
||||
@ApiModelProperty(value = "是否答对", required=true)
|
||||
private Boolean isRight;
|
||||
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
package com.guwan.backend.model.paper.dto.ext;
|
||||
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题备选答案请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷考题备选答案", description="试卷考题备选答案")
|
||||
public class PaperQuAnswerExtDTO extends PaperQuAnswerDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
@ApiModelProperty(value = "试题图片", required=true)
|
||||
private String image;
|
||||
|
||||
@ApiModelProperty(value = "答案内容", required=true)
|
||||
private String content;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.paper.dto.ext;
|
||||
|
||||
|
||||
import com.guwan.backend.model.paper.dto.PaperQuDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷题目详情类", description="试卷题目详情类")
|
||||
public class PaperQuDetailDTO extends PaperQuDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
@ApiModelProperty(value = "图片", required=true)
|
||||
private String image;
|
||||
|
||||
@ApiModelProperty(value = "题目内容", required=true)
|
||||
private String content;
|
||||
|
||||
@ApiModelProperty(value = "答案内容", required=true)
|
||||
List<PaperQuAnswerExtDTO> answerList;
|
||||
}
|
|
@ -0,0 +1,22 @@
|
|||
package com.guwan.backend.model.paper.dto.request;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* @author bool
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="查找试卷题目详情请求类", description="查找试卷题目详情请求类")
|
||||
public class PaperAnswerDTO extends PaperQuQueryDTO {
|
||||
|
||||
@ApiModelProperty(value = "回答列表", required=true)
|
||||
private List<String> answers;
|
||||
|
||||
@ApiModelProperty(value = "主观答案", required=true)
|
||||
private String answer;
|
||||
|
||||
}
|
|
@ -0,0 +1,22 @@
|
|||
package com.guwan.backend.model.paper.dto.request;
|
||||
|
||||
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||
import com.yf.exam.core.api.dto.BaseDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* @author bool
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷创建请求类", description="试卷创建请求类")
|
||||
public class PaperCreateReqDTO extends BaseDTO {
|
||||
|
||||
@JsonIgnore
|
||||
private String userId;
|
||||
|
||||
@ApiModelProperty(value = "考试ID", required=true)
|
||||
private String examId;
|
||||
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
package com.guwan.backend.model.paper.dto.request;
|
||||
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.io.Serializable;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷", description="试卷")
|
||||
public class PaperListReqDTO implements Serializable {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
@ApiModelProperty(value = "用户ID", required=true)
|
||||
private String userId;
|
||||
|
||||
@ApiModelProperty(value = "部门ID", required=true)
|
||||
private String departId;
|
||||
|
||||
@ApiModelProperty(value = "规则ID", required=true)
|
||||
private String examId;
|
||||
|
||||
@ApiModelProperty(value = "用户昵称", required=true)
|
||||
private String realName;
|
||||
|
||||
@ApiModelProperty(value = "试卷状态", required=true)
|
||||
private Integer state;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,21 @@
|
|||
package com.guwan.backend.model.paper.dto.request;
|
||||
|
||||
import com.yf.exam.core.api.dto.BaseDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* @author bool
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="查找试卷题目详情请求类", description="查找试卷题目详情请求类")
|
||||
public class PaperQuQueryDTO extends BaseDTO {
|
||||
|
||||
@ApiModelProperty(value = "试卷ID", required=true)
|
||||
private String paperId;
|
||||
|
||||
@ApiModelProperty(value = "题目ID", required=true)
|
||||
private String quId;
|
||||
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
package com.guwan.backend.model.paper.dto.response;
|
||||
|
||||
|
||||
import com.guwan.backend.model.paper.dto.PaperDTO;
|
||||
import com.guwan.backend.model.paper.dto.PaperQuDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.Calendar;
|
||||
import java.util.List;
|
||||
|
||||
@Data
|
||||
@ApiModel(value="考试详情", description="考试详情")
|
||||
public class ExamDetailRespDTO extends PaperDTO {
|
||||
|
||||
|
||||
@ApiModelProperty(value = "单选题列表", required=true)
|
||||
private List<PaperQuDTO> radioList;
|
||||
|
||||
@ApiModelProperty(value = "多选题列表", required=true)
|
||||
private List<PaperQuDTO> multiList;
|
||||
|
||||
@ApiModelProperty(value = "判断题", required=true)
|
||||
private List<PaperQuDTO> judgeList;
|
||||
|
||||
|
||||
@ApiModelProperty(value = "剩余结束秒数", required=true)
|
||||
public Long getLeftSeconds(){
|
||||
|
||||
// 结束时间
|
||||
Calendar cl = Calendar.getInstance();
|
||||
cl.setTime(this.getCreateTime());
|
||||
cl.add(Calendar.MINUTE, getTotalTime());
|
||||
|
||||
return (cl.getTimeInMillis() - System.currentTimeMillis()) / 1000;
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,19 @@
|
|||
package com.guwan.backend.model.paper.dto.response;
|
||||
|
||||
|
||||
import com.guwan.backend.model.paper.dto.PaperDTO;
|
||||
import com.guwan.backend.model.paper.dto.ext.PaperQuDetailDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
@Data
|
||||
@ApiModel(value="考试结果展示响应类", description="考试结果展示响应类")
|
||||
public class ExamResultRespDTO extends PaperDTO {
|
||||
|
||||
@ApiModelProperty(value = "问题列表", required=true)
|
||||
private List<PaperQuDetailDTO> quList;
|
||||
|
||||
}
|
|
@ -0,0 +1,27 @@
|
|||
package com.guwan.backend.model.paper.dto.response;
|
||||
|
||||
|
||||
import com.guwan.backend.model.paper.dto.PaperDTO;
|
||||
import io.swagger.annotations.ApiModel;
|
||||
import io.swagger.annotations.ApiModelProperty;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷请求类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@ApiModel(value="试卷列表响应类", description="试卷列表响应类")
|
||||
public class PaperListRespDTO extends PaperDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
@ApiModelProperty(value = "人员", required=true)
|
||||
private String realName;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,125 @@
|
|||
package com.guwan.backend.model.paper.entity;
|
||||
|
||||
import com.baomidou.mybatisplus.annotation.IdType;
|
||||
import com.baomidou.mybatisplus.annotation.TableField;
|
||||
import com.baomidou.mybatisplus.annotation.TableId;
|
||||
import com.baomidou.mybatisplus.annotation.TableName;
|
||||
import com.baomidou.mybatisplus.extension.activerecord.Model;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.Date;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷实体类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@TableName("el_paper")
|
||||
public class Paper extends Model<Paper> {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/**
|
||||
* 试卷ID
|
||||
*/
|
||||
@TableId(value = "id", type = IdType.ASSIGN_ID)
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* 用户ID
|
||||
*/
|
||||
@TableField("user_id")
|
||||
private String userId;
|
||||
|
||||
/**
|
||||
* 部门ID
|
||||
*/
|
||||
@TableField("depart_id")
|
||||
private String departId;
|
||||
|
||||
/**
|
||||
* 规则ID
|
||||
*/
|
||||
@TableField("exam_id")
|
||||
private String examId;
|
||||
|
||||
/**
|
||||
* 考试标题
|
||||
*/
|
||||
private String title;
|
||||
|
||||
/**
|
||||
* 考试时长
|
||||
*/
|
||||
@TableField("total_time")
|
||||
private Integer totalTime;
|
||||
|
||||
/**
|
||||
* 用户时长
|
||||
*/
|
||||
@TableField("user_time")
|
||||
private Integer userTime;
|
||||
|
||||
/**
|
||||
* 试卷总分
|
||||
*/
|
||||
@TableField("total_score")
|
||||
private Integer totalScore;
|
||||
|
||||
/**
|
||||
* 及格分
|
||||
*/
|
||||
@TableField("qualify_score")
|
||||
private Integer qualifyScore;
|
||||
|
||||
/**
|
||||
* 客观分
|
||||
*/
|
||||
@TableField("obj_score")
|
||||
private Integer objScore;
|
||||
|
||||
/**
|
||||
* 主观分
|
||||
*/
|
||||
@TableField("subj_score")
|
||||
private Integer subjScore;
|
||||
|
||||
/**
|
||||
* 用户得分
|
||||
*/
|
||||
@TableField("user_score")
|
||||
private Integer userScore;
|
||||
|
||||
/**
|
||||
* 是否包含简答题
|
||||
*/
|
||||
@TableField("has_saq")
|
||||
private Boolean hasSaq;
|
||||
|
||||
/**
|
||||
* 试卷状态
|
||||
*/
|
||||
private Integer state;
|
||||
|
||||
/**
|
||||
* 创建时间
|
||||
*/
|
||||
@TableField("create_time")
|
||||
private Date createTime;
|
||||
|
||||
/**
|
||||
* 更新时间
|
||||
*/
|
||||
@TableField("update_time")
|
||||
private Date updateTime;
|
||||
|
||||
/**
|
||||
* 截止时间
|
||||
*/
|
||||
@TableField("limit_time")
|
||||
private Date limitTime;
|
||||
}
|
|
@ -0,0 +1,80 @@
|
|||
package com.guwan.backend.model.paper.entity;
|
||||
|
||||
import com.baomidou.mybatisplus.annotation.IdType;
|
||||
import com.baomidou.mybatisplus.annotation.TableField;
|
||||
import com.baomidou.mybatisplus.annotation.TableId;
|
||||
import com.baomidou.mybatisplus.annotation.TableName;
|
||||
import com.baomidou.mybatisplus.extension.activerecord.Model;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题实体类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@TableName("el_paper_qu")
|
||||
public class PaperQu extends Model<PaperQu> {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/**
|
||||
* ID
|
||||
*/
|
||||
@TableId(value = "id", type = IdType.ASSIGN_ID)
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* 试卷ID
|
||||
*/
|
||||
@TableField("paper_id")
|
||||
private String paperId;
|
||||
|
||||
/**
|
||||
* 题目ID
|
||||
*/
|
||||
@TableField("qu_id")
|
||||
private String quId;
|
||||
|
||||
/**
|
||||
* 题目类型
|
||||
*/
|
||||
@TableField("qu_type")
|
||||
private Integer quType;
|
||||
|
||||
/**
|
||||
* 是否已答
|
||||
*/
|
||||
private Boolean answered;
|
||||
|
||||
/**
|
||||
* 主观答案
|
||||
*/
|
||||
private String answer;
|
||||
|
||||
/**
|
||||
* 问题排序
|
||||
*/
|
||||
private Integer sort;
|
||||
|
||||
/**
|
||||
* 单题分分值
|
||||
*/
|
||||
private Integer score;
|
||||
|
||||
/**
|
||||
* 实际得分(主观题)
|
||||
*/
|
||||
@TableField("actual_score")
|
||||
private Integer actualScore;
|
||||
|
||||
/**
|
||||
* 是否答对
|
||||
*/
|
||||
@TableField("is_right")
|
||||
private Boolean isRight;
|
||||
|
||||
}
|
|
@ -0,0 +1,68 @@
|
|||
package com.guwan.backend.model.paper.entity;
|
||||
|
||||
import com.baomidou.mybatisplus.annotation.IdType;
|
||||
import com.baomidou.mybatisplus.annotation.TableField;
|
||||
import com.baomidou.mybatisplus.annotation.TableId;
|
||||
import com.baomidou.mybatisplus.annotation.TableName;
|
||||
import com.baomidou.mybatisplus.extension.activerecord.Model;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题备选答案实体类
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 17:31
|
||||
*/
|
||||
@Data
|
||||
@TableName("el_paper_qu_answer")
|
||||
public class PaperQuAnswer extends Model<PaperQuAnswer> {
|
||||
|
||||
|
||||
/**
|
||||
* 自增ID
|
||||
*/
|
||||
@TableId(value = "id", type = IdType.ASSIGN_ID)
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* 试卷ID
|
||||
*/
|
||||
@TableField("paper_id")
|
||||
private String paperId;
|
||||
|
||||
/**
|
||||
* 回答项ID
|
||||
*/
|
||||
@TableField("answer_id")
|
||||
private String answerId;
|
||||
|
||||
/**
|
||||
* 题目ID
|
||||
*/
|
||||
@TableField("qu_id")
|
||||
private String quId;
|
||||
|
||||
/**
|
||||
* 是否正确项
|
||||
*/
|
||||
@TableField("is_right")
|
||||
private Boolean isRight;
|
||||
|
||||
/**
|
||||
* 是否选中
|
||||
*/
|
||||
private Boolean checked;
|
||||
|
||||
/**
|
||||
* 排序
|
||||
*/
|
||||
private Integer sort;
|
||||
|
||||
/**
|
||||
* 选项标签
|
||||
*/
|
||||
private String abc;
|
||||
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.paper.enums;
|
||||
|
||||
|
||||
/**
|
||||
* 考试状态
|
||||
* @author bool
|
||||
* @date 2019-10-30 13:11
|
||||
*/
|
||||
public interface ExamState {
|
||||
|
||||
|
||||
/**
|
||||
* 考试中
|
||||
*/
|
||||
Integer ENABLE = 0;
|
||||
|
||||
/**
|
||||
* 待阅卷
|
||||
*/
|
||||
Integer DISABLED = 1;
|
||||
|
||||
/**
|
||||
* 已完成
|
||||
*/
|
||||
Integer READY_START = 2;
|
||||
|
||||
/**
|
||||
* 已结束
|
||||
*/
|
||||
Integer OVERDUE = 3;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package com.guwan.backend.model.paper.enums;
|
||||
|
||||
|
||||
/**
|
||||
* 试卷状态
|
||||
* @author bool
|
||||
* @date 2019-10-30 13:11
|
||||
*/
|
||||
public interface PaperState {
|
||||
|
||||
|
||||
/**
|
||||
* 考试中
|
||||
*/
|
||||
Integer ING = 0;
|
||||
|
||||
/**
|
||||
* 待阅卷
|
||||
*/
|
||||
Integer WAIT_OPT = 1;
|
||||
|
||||
/**
|
||||
* 已完成
|
||||
*/
|
||||
Integer FINISHED = 2;
|
||||
|
||||
/**
|
||||
* 弃考
|
||||
*/
|
||||
Integer BREAK = 3;
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,45 @@
|
|||
package com.guwan.backend.model.paper.job;
|
||||
|
||||
import com.yf.exam.ability.job.service.JobService;
|
||||
import com.yf.exam.modules.paper.service.PaperService;
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.quartz.Job;
|
||||
import org.quartz.JobDetail;
|
||||
import org.quartz.JobExecutionContext;
|
||||
import org.quartz.JobExecutionException;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
/**
|
||||
* 超时自动交卷任务
|
||||
* @author bool
|
||||
*/
|
||||
@Log4j2
|
||||
@Component
|
||||
public class BreakExamJob implements Job {
|
||||
|
||||
@Autowired
|
||||
private PaperService paperService;
|
||||
|
||||
@Override
|
||||
public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {
|
||||
|
||||
JobDetail detail = jobExecutionContext.getJobDetail();
|
||||
String name = detail.getKey().getName();
|
||||
String group = detail.getKey().getGroup();
|
||||
String data = String.valueOf(detail.getJobDataMap().get(JobService.TASK_DATA));
|
||||
|
||||
log.info("++++++++++定时任务:处理到期的交卷");
|
||||
log.info("++++++++++jobName:{}", name);
|
||||
log.info("++++++++++jobGroup:{}", group);
|
||||
log.info("++++++++++taskData:{}", data);
|
||||
|
||||
|
||||
// 强制交卷
|
||||
paperService.handExam(data);
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,40 @@
|
|||
package com.guwan.backend.model.paper.mapper;
|
||||
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||
|
||||
import com.guwan.backend.model.paper.dto.PaperDTO;
|
||||
import com.guwan.backend.model.paper.dto.request.PaperListReqDTO;
|
||||
import com.guwan.backend.model.paper.dto.response.PaperListRespDTO;
|
||||
import com.guwan.backend.model.paper.entity.Paper;
|
||||
import org.apache.ibatis.annotations.Param;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷Mapper
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 16:33
|
||||
*/
|
||||
public interface PaperMapper extends BaseMapper<Paper> {
|
||||
|
||||
/**
|
||||
* 查找试卷分页
|
||||
* @param page
|
||||
* @param query
|
||||
* @return
|
||||
*/
|
||||
IPage<PaperListRespDTO> paging(Page page, @Param("query") PaperListReqDTO query);
|
||||
|
||||
|
||||
/**
|
||||
* 试卷列表响应类
|
||||
* @param query
|
||||
* @return
|
||||
*/
|
||||
List<PaperListRespDTO> list(@Param("query") PaperDTO query);
|
||||
}
|
|
@ -0,0 +1,27 @@
|
|||
package com.guwan.backend.model.paper.mapper;
|
||||
|
||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||
import com.guwan.backend.model.paper.dto.ext.PaperQuAnswerExtDTO;
|
||||
import com.guwan.backend.model.paper.entity.PaperQuAnswer;
|
||||
import org.apache.ibatis.annotations.Param;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* 试卷考题备选答案Mapper
|
||||
* </p>
|
||||
*
|
||||
* @author 聪明笨狗
|
||||
* @since 2020-05-25 16:33
|
||||
*/
|
||||
public interface PaperQuAnswerMapper extends BaseMapper<PaperQuAnswer> {
|
||||
|
||||
/**
|
||||
* 查找试卷试题答案列表
|
||||
* @param paperId
|
||||
* @param quId
|
||||
* @return
|
||||
*/
|
||||
List<PaperQuAnswerExtDTO> list(@Param("paperId") String paperId, @Param("quId") String quId);
|
||||
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue