deason il y a 5 ans
commit
4fc9f10ff5
92 fichiers modifiés avec 6604 ajouts et 0 suppressions
  1. 12 0
      .gitignore
  2. 30 0
      assembly.xml
  3. 4 0
      jenkins-dev.sh
  4. 4 0
      jenkins-prod.sh
  5. 4 0
      jenkins-test.sh
  6. 243 0
      pom.xml
  7. 19 0
      shell/jenkins.sh
  8. 1 0
      shell/start.args
  9. 34 0
      shell/start.sh
  10. 1 0
      shell/start.vmoptions
  11. 18 0
      shell/stop.sh
  12. 1 0
      shell/version
  13. 43 0
      src/main/java/cn/com/qmth/examcloud/app/ApiApplication.java
  14. 62 0
      src/main/java/cn/com/qmth/examcloud/app/controller/DeviceRecordController.java
  15. 65 0
      src/main/java/cn/com/qmth/examcloud/app/controller/ExamController.java
  16. 29 0
      src/main/java/cn/com/qmth/examcloud/app/controller/IndexController.java
  17. 102 0
      src/main/java/cn/com/qmth/examcloud/app/controller/OfflineExamController.java
  18. 35 0
      src/main/java/cn/com/qmth/examcloud/app/controller/PaperController.java
  19. 178 0
      src/main/java/cn/com/qmth/examcloud/app/controller/PracticeExamRestController.java
  20. 48 0
      src/main/java/cn/com/qmth/examcloud/app/controller/RouterController.java
  21. 65 0
      src/main/java/cn/com/qmth/examcloud/app/controller/SystemRestController.java
  22. 129 0
      src/main/java/cn/com/qmth/examcloud/app/controller/UserAuthRestController.java
  23. 74 0
      src/main/java/cn/com/qmth/examcloud/app/core/CloudDiscoveryClient.java
  24. 43 0
      src/main/java/cn/com/qmth/examcloud/app/core/Constants.java
  25. 64 0
      src/main/java/cn/com/qmth/examcloud/app/core/SysProperty.java
  26. 127 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/AccessInterceptor.java
  27. 38 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/ControllerAdviceHandler.java
  28. 66 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/CustomHttpServletRequest.java
  29. 42 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/FilterConfig.java
  30. 36 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/InterceptorConfig.java
  31. 39 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/RedirectViewResolver.java
  32. 44 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/RedisConfig.java
  33. 44 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/SwaggerConfig.java
  34. 255 0
      src/main/java/cn/com/qmth/examcloud/app/core/config/TokenFilter.java
  35. 21 0
      src/main/java/cn/com/qmth/examcloud/app/core/exception/ApiException.java
  36. 7 0
      src/main/java/cn/com/qmth/examcloud/app/core/router/Method.java
  37. 98 0
      src/main/java/cn/com/qmth/examcloud/app/core/router/Router.java
  38. 40 0
      src/main/java/cn/com/qmth/examcloud/app/core/router/Server.java
  39. 41 0
      src/main/java/cn/com/qmth/examcloud/app/core/utils/DateUtils.java
  40. 46 0
      src/main/java/cn/com/qmth/examcloud/app/core/utils/HttpClientBuilder.java
  41. 169 0
      src/main/java/cn/com/qmth/examcloud/app/core/utils/HttpUtils.java
  42. 224 0
      src/main/java/cn/com/qmth/examcloud/app/core/utils/JsonMapper.java
  43. 55 0
      src/main/java/cn/com/qmth/examcloud/app/core/utils/StrUtils.java
  44. 94 0
      src/main/java/cn/com/qmth/examcloud/app/core/utils/ThreadUtils.java
  45. 20 0
      src/main/java/cn/com/qmth/examcloud/app/dao/DeviceRecordRepository.java
  46. 73 0
      src/main/java/cn/com/qmth/examcloud/app/model/Constants.java
  47. 178 0
      src/main/java/cn/com/qmth/examcloud/app/model/DeviceRecord.java
  48. 42 0
      src/main/java/cn/com/qmth/examcloud/app/model/DeviceRecordQuery.java
  49. 27 0
      src/main/java/cn/com/qmth/examcloud/app/model/ExamInfo.java
  50. 68 0
      src/main/java/cn/com/qmth/examcloud/app/model/GetYunSignatureReq.java
  51. 31 0
      src/main/java/cn/com/qmth/examcloud/app/model/IdEntity.java
  52. 194 0
      src/main/java/cn/com/qmth/examcloud/app/model/LoginInfo.java
  53. 38 0
      src/main/java/cn/com/qmth/examcloud/app/model/LoginType.java
  54. 74 0
      src/main/java/cn/com/qmth/examcloud/app/model/ResBody.java
  55. 119 0
      src/main/java/cn/com/qmth/examcloud/app/model/Result.java
  56. 223 0
      src/main/java/cn/com/qmth/examcloud/app/model/UserInfo.java
  57. 140 0
      src/main/java/cn/com/qmth/examcloud/app/service/CoreAuthService.java
  58. 30 0
      src/main/java/cn/com/qmth/examcloud/app/service/CoreBasicService.java
  59. 75 0
      src/main/java/cn/com/qmth/examcloud/app/service/CoreExamWorkService.java
  60. 309 0
      src/main/java/cn/com/qmth/examcloud/app/service/CoreOeService.java
  61. 51 0
      src/main/java/cn/com/qmth/examcloud/app/service/CoreQuestionService.java
  62. 40 0
      src/main/java/cn/com/qmth/examcloud/app/service/DeviceRecordService.java
  63. 82 0
      src/main/java/cn/com/qmth/examcloud/app/service/RedisService.java
  64. 147 0
      src/main/java/cn/com/qmth/examcloud/app/service/RouterService.java
  65. 27 0
      src/main/java/cn/com/qmth/examcloud/app/service/UpYunService.java
  66. 326 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreAuthServiceImpl.java
  67. 39 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreBasicServiceImpl.java
  68. 96 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreExamWorkServiceImpl.java
  69. 308 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreOeServiceImpl.java
  70. 92 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreQuestionServiceImpl.java
  71. 107 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/DeviceRecordServiceImpl.java
  72. 35 0
      src/main/java/cn/com/qmth/examcloud/app/service/impl/UpYunServiceImpl.java
  73. 6 0
      src/main/resources/application.properties
  74. 1 0
      src/main/resources/classpath.location
  75. 82 0
      src/main/resources/log4j2.xml
  76. 5 0
      src/main/resources/security-exclusions.conf
  77. 0 0
      src/main/resources/security.properties
  78. 99 0
      src/main/resources/static/api/app-api/page.js
  79. 4 0
      src/main/resources/static/api/app-api/styles/bootstrap/css/bootstrap.min.css
  80. 0 0
      src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.eot
  81. 288 0
      src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.svg
  82. 0 0
      src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.ttf
  83. 0 0
      src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.woff
  84. 0 0
      src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.woff2
  85. 5 0
      src/main/resources/static/api/app-api/styles/bootstrap/js/bootstrap.min.js
  86. 1 0
      src/main/resources/static/api/app-api/styles/jquery/jquery-1.9.1.min.js
  87. 35 0
      src/main/resources/static/api/app-api/upload.html
  88. 191 0
      src/main/resources/templates/deviceRecord/list.ftl
  89. 37 0
      src/main/resources/templates/error.ftl
  90. 15 0
      src/main/resources/templates/index.ftl
  91. 34 0
      src/test/java/cn/com/qmth/examcloud/app/ServiceTest.java
  92. 16 0
      src/test/java/cn/com/qmth/examcloud/app/SimpleTest.java

+ 12 - 0
.gitignore

@@ -0,0 +1,12 @@
+*.class
+.project
+.classpath
+.settings
+.idea/
+*.iml
+*.log
+*.jar
+*.war
+*.ear
+logs/
+target/

+ 30 - 0
assembly.xml

@@ -0,0 +1,30 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<assembly xmlns="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.2" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+	xsi:schemaLocation="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.2 http://maven.apache.org/xsd/assembly-1.1.2.xsd">
+	<id>distribution</id>
+	<formats>
+		<format>zip</format>
+	</formats>
+	<fileSets>
+		<fileSet>
+			<directory>${project.basedir}/src/main/resources</directory>
+			<outputDirectory>/config</outputDirectory>
+		</fileSet>
+		<fileSet>
+			<directory>${project.basedir}/shell</directory>
+			<excludes>
+				<exclude>start.args</exclude>
+				<exclude>start.vmoptions</exclude>
+			</excludes>
+			<outputDirectory>/</outputDirectory>
+			<fileMode>0777</fileMode>
+		</fileSet>
+	</fileSets>
+	<dependencySets>
+		<dependencySet>
+			<useProjectArtifact>true</useProjectArtifact>
+			<outputDirectory>lib</outputDirectory>
+			<scope>runtime</scope>
+		</dependencySet>
+	</dependencySets>
+</assembly>

+ 4 - 0
jenkins-dev.sh

@@ -0,0 +1,4 @@
+#!/bin/bash
+
+pwd
+cp target/examcloud-app-api-distribution.zip ~/packages

+ 4 - 0
jenkins-prod.sh

@@ -0,0 +1,4 @@
+#!/bin/bash
+
+pwd
+cp target/examcloud-app-api-distribution.zip ~/packages

+ 4 - 0
jenkins-test.sh

@@ -0,0 +1,4 @@
+#!/bin/bash
+
+pwd
+cp target/examcloud-app-api-distribution.zip ~/packages

+ 243 - 0
pom.xml

@@ -0,0 +1,243 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
+    <modelVersion>4.0.0</modelVersion>
+    <artifactId>examcloud-app-api</artifactId>
+    <version>2019-SNAPSHOT</version>
+    <packaging>jar</packaging>
+
+    <parent>
+        <groupId>cn.com.qmth.examcloud</groupId>
+        <artifactId>examcloud-parent</artifactId>
+        <version>2019</version>
+    </parent>
+
+    <properties>
+        <examcloud.version>2019-SNAPSHOT</examcloud.version>
+        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
+        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
+        <java.version>1.8</java.version>
+    </properties>
+
+    <dependencies>
+        <dependency>
+            <groupId>cn.com.qmth.examcloud</groupId>
+            <artifactId>examcloud-support</artifactId>
+            <version>${examcloud.version}</version>
+        </dependency>
+        <dependency>
+            <groupId>cn.com.qmth.examcloud.rpc</groupId>
+            <artifactId>examcloud-global-api</artifactId>
+            <version>${examcloud.version}</version>
+        </dependency>
+        <dependency>
+            <groupId>cn.com.qmth.examcloud.rpc</groupId>
+            <artifactId>examcloud-core-basic-api-client</artifactId>
+            <version>${examcloud.version}</version>
+        </dependency>
+        <dependency>
+            <groupId>com.squareup.okhttp3</groupId>
+            <artifactId>okhttp</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>com.alibaba</groupId>
+            <artifactId>fastjson</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>com.google.guava</groupId>
+            <artifactId>guava</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>org.apache.commons</groupId>
+            <artifactId>commons-lang3</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>commons-collections</groupId>
+            <artifactId>commons-collections</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>commons-codec</groupId>
+            <artifactId>commons-codec</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>org.assertj</groupId>
+            <artifactId>assertj-core</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>io.springfox</groupId>
+            <artifactId>springfox-swagger2</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>com.github.xiaoymin</groupId>
+            <artifactId>swagger-bootstrap-ui</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>org.springframework.boot</groupId>
+            <artifactId>spring-boot-starter-freemarker</artifactId>
+            <exclusions>
+                <exclusion>
+                    <groupId>org.springframework.boot</groupId>
+                    <artifactId>spring-boot-starter-logging</artifactId>
+                </exclusion>
+            </exclusions>
+        </dependency>
+        <dependency>
+            <groupId>org.springframework.boot</groupId>
+            <artifactId>spring-boot-starter-log4j2</artifactId>
+        </dependency>
+        <dependency>
+            <groupId>org.springframework.boot</groupId>
+            <artifactId>spring-boot-starter-test</artifactId>
+            <scope>test</scope>
+            <exclusions>
+                <exclusion>
+                    <groupId>org.springframework.boot</groupId>
+                    <artifactId>spring-boot-starter-logging</artifactId>
+                </exclusion>
+            </exclusions>
+        </dependency>
+    </dependencies>
+
+    <dependencyManagement>
+        <dependencies>
+            <dependency>
+                <groupId>com.squareup.okhttp3</groupId>
+                <artifactId>okhttp</artifactId>
+                <version>4.2.2</version>
+            </dependency>
+            <dependency>
+                <groupId>com.alibaba</groupId>
+                <artifactId>fastjson</artifactId>
+                <version>1.2.62</version>
+            </dependency>
+            <dependency>
+                <groupId>com.google.guava</groupId>
+                <artifactId>guava</artifactId>
+                <version>28.1-jre</version>
+            </dependency>
+            <dependency>
+                <groupId>org.apache.commons</groupId>
+                <artifactId>commons-lang3</artifactId>
+                <version>3.9</version>
+            </dependency>
+            <dependency>
+                <groupId>org.apache.commons</groupId>
+                <artifactId>commons-compress</artifactId>
+                <version>1.19</version>
+            </dependency>
+            <dependency>
+                <groupId>commons-collections</groupId>
+                <artifactId>commons-collections</artifactId>
+                <version>3.2.2</version>
+            </dependency>
+            <dependency>
+                <groupId>commons-codec</groupId>
+                <artifactId>commons-codec</artifactId>
+                <version>1.13</version>
+            </dependency>
+            <dependency>
+                <groupId>com.google.code.findbugs</groupId>
+                <artifactId>jsr305</artifactId>
+                <version>3.0.2</version>
+            </dependency>
+            <dependency>
+                <groupId>org.javassist</groupId>
+                <artifactId>javassist</artifactId>
+                <version>3.26.0-GA</version>
+            </dependency>
+            <dependency>
+                <groupId>com.mchange</groupId>
+                <artifactId>mchange-commons-java</artifactId>
+                <version>0.2.19</version>
+            </dependency>
+            <dependency>
+                <groupId>org.assertj</groupId>
+                <artifactId>assertj-core</artifactId>
+                <version>3.14.0</version>
+            </dependency>
+            <dependency>
+                <groupId>io.swagger</groupId>
+                <artifactId>swagger-annotations</artifactId>
+                <version>1.5.24</version>
+            </dependency>
+            <dependency>
+                <groupId>io.swagger</groupId>
+                <artifactId>swagger-models</artifactId>
+                <version>1.5.24</version>
+            </dependency>
+            <dependency>
+                <groupId>io.springfox</groupId>
+                <artifactId>springfox-swagger2</artifactId>
+                <version>2.9.2</version>
+            </dependency>
+            <dependency>
+                <groupId>com.github.xiaoymin</groupId>
+                <artifactId>swagger-bootstrap-ui</artifactId>
+                <version>1.9.6</version>
+            </dependency>
+        </dependencies>
+    </dependencyManagement>
+
+    <build>
+        <finalName>examcloud-app-api</finalName>
+
+        <plugins>
+            <plugin>
+                <groupId>org.springframework.boot</groupId>
+                <artifactId>spring-boot-maven-plugin</artifactId>
+            </plugin>
+
+            <plugin>
+                <groupId>org.apache.maven.plugins</groupId>
+                <artifactId>maven-surefire-plugin</artifactId>
+                <configuration>
+                    <skip>true</skip>
+                    <skipTests>true</skipTests>
+                </configuration>
+            </plugin>
+
+            <plugin>
+                <groupId>org.apache.maven.plugins</groupId>
+                <artifactId>maven-jar-plugin</artifactId>
+                <configuration>
+                    <archive>
+                        <manifest>
+                            <mainClass>cn.com.qmth.examcloud.app.ApiApplication</mainClass>
+                            <addClasspath>true</addClasspath>
+                            <classpathPrefix>./</classpathPrefix>
+                        </manifest>
+                        <manifestEntries>
+                            <Class-Path>../config/</Class-Path>
+                        </manifestEntries>
+                    </archive>
+                    <excludes>
+                        <exclude>templates/*</exclude>
+                        <exclude>*.properties</exclude>
+                        <exclude>*.xml</exclude>
+                        <exclude>*.conf</exclude>
+                    </excludes>
+                </configuration>
+            </plugin>
+
+            <plugin>
+                <groupId>org.apache.maven.plugins</groupId>
+                <artifactId>maven-assembly-plugin</artifactId>
+                <configuration>
+                    <finalName>examcloud-app-api</finalName>
+                    <descriptors>
+                        <descriptor>assembly.xml</descriptor>
+                    </descriptors>
+                </configuration>
+                <executions>
+                    <execution>
+                        <id>make-assembly</id>
+                        <phase>install</phase>
+                        <goals>
+                            <goal>assembly</goal>
+                        </goals>
+                    </execution>
+                </executions>
+            </plugin>
+        </plugins>
+    </build>
+</project>

+ 19 - 0
shell/jenkins.sh

@@ -0,0 +1,19 @@
+#!/bin/bash
+pwd
+
+rm -rf /home/qmth/project/examcloud/examcloud-app-api-distribution.zip
+rm -rf /home/qmth/project/examcloud/examcloud-app-api/lib
+rm -rf /home/qmth/project/examcloud/examcloud-app-api/config
+
+cp target/examcloud-app-api-distribution.zip ~/project/examcloud/
+
+cd  ~/project/examcloud/
+unzip -o -q examcloud-app-api-distribution.zip
+
+cd examcloud-app-api
+echo "--spring.profiles.active=test --examcloud.startup.configCenterHost=localhost" > start.args
+echo "-server -Xms512m -Xmx512m  -XX:-UseGCOverheadLimit" > start.vmoptions
+
+bash stop.sh
+BUILD_ID=DONTKILLME
+bash start.sh 110

+ 1 - 0
shell/start.args

@@ -0,0 +1 @@
+--spring.profiles.active=test

+ 34 - 0
shell/start.sh

@@ -0,0 +1,34 @@
+#!/bin/bash
+
+FILE_PATH=$(cd `dirname $0`; pwd)
+APP_VERSION=`cat $FILE_PATH/version`
+APP_MAIN_JAR="examcloud-app-api-"$APP_VERSION"-SNAPSHOT.jar"
+JAVA_OPTS=`cat $FILE_PATH/start.vmoptions`
+APP_ARGS=`cat $FILE_PATH/start.args`
+
+PID_LIST=`ps -ef|grep $APP_MAIN_JAR|grep java|awk '{print $2}'`
+
+if [ ! -z "$PID_LIST" ]; then
+    echo "[ERROR] : APP is already running!"
+    exit -1
+fi
+
+if [ "$1" ];then
+    echo "startupCode:"$1;
+else
+    echo "[ERROR] : no arguments"
+    exit -1
+fi
+
+APP_ARGS=$APP_ARGS" --examcloud.startup.startupCode="$1
+
+echo "java options:"
+echo "$JAVA_OPTS"
+echo "args:"
+echo "$APP_ARGS"
+    
+nohup java $JAVA_OPTS -jar $FILE_PATH/lib/$APP_MAIN_JAR $APP_ARGS >/dev/null 2>&1 &
+
+echo "starting......"
+
+exit 0

+ 1 - 0
shell/start.vmoptions

@@ -0,0 +1 @@
+-server -Xms512m -Xmx512m

+ 18 - 0
shell/stop.sh

@@ -0,0 +1,18 @@
+#!/bin/bash
+
+FILE_PATH=$(cd `dirname $0`; pwd)
+APP_VERSION=`cat $FILE_PATH/version`
+APP_MAIN_JAR="examcloud-app-api-"$APP_VERSION"-SNAPSHOT.jar"
+
+PID_LIST=`ps -ef|grep $APP_MAIN_JAR|grep java|awk '{print $2}'`
+
+if [ ! -z "$PID_LIST" ]; then
+    echo "Runnable jar is $APP_MAIN_JAR."
+    for PID in $PID_LIST 
+    do
+        kill -9 $PID
+    done
+    echo "stopped !"
+fi
+
+exit 0

+ 1 - 0
shell/version

@@ -0,0 +1 @@
+2019

+ 43 - 0
src/main/java/cn/com/qmth/examcloud/app/ApiApplication.java

@@ -0,0 +1,43 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 15:00:21.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app;
+
+import cn.com.qmth.examcloud.web.bootstrap.AppBootstrap;
+import org.springframework.boot.autoconfigure.SpringBootApplication;
+import org.springframework.boot.autoconfigure.domain.EntityScan;
+import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
+import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
+import org.springframework.context.annotation.ComponentScan;
+import org.springframework.data.jpa.repository.config.EnableJpaAuditing;
+import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
+import org.springframework.scheduling.annotation.EnableAsync;
+
+@EnableAsync
+@EnableJpaAuditing
+@EnableEurekaClient
+@EnableDiscoveryClient
+@SpringBootApplication
+@ComponentScan(basePackages = {"cn.com.qmth.examcloud"})
+@EntityScan(basePackages = {"cn.com.qmth.examcloud"})
+@EnableJpaRepositories(basePackages = {"cn.com.qmth.examcloud"})
+public class ApiApplication {
+
+    static {
+        String runtimeLevel = System.getProperty("log.commonLevel");
+        if (null == runtimeLevel) {
+            System.setProperty("log.commonLevel", "INFO");
+        }
+        System.setProperty("hibernate.dialect.storage_engine", "innodb");
+    }
+
+    public static void main(String[] args) {
+        //SpringApplication.run(ApiApplication.class, args);
+        AppBootstrap.run(ApiApplication.class, args);
+    }
+
+}

+ 62 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/DeviceRecordController.java

@@ -0,0 +1,62 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 11:04:17.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import cn.com.qmth.examcloud.app.model.DeviceRecord;
+import cn.com.qmth.examcloud.app.model.DeviceRecordQuery;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreAuthService;
+import cn.com.qmth.examcloud.app.service.DeviceRecordService;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.data.domain.Page;
+import org.springframework.stereotype.Controller;
+import org.springframework.ui.Model;
+import org.springframework.web.bind.annotation.*;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_TOKEN;
+
+/**
+ * 设备访问记录Controller
+ */
+@Controller
+@RequestMapping("${$rmp}/device/record")
+public class DeviceRecordController {
+    private static Logger log = LoggerFactory.getLogger(DeviceRecordController.class);
+    @Autowired
+    private DeviceRecordService deviceRecordService;
+    @Autowired
+    private CoreAuthService authService;
+
+    @RequestMapping(value = "/list", method = RequestMethod.GET)
+    public String list(@RequestParam String key, @RequestParam String token, Model model) throws Exception {
+        model.addAttribute(PARAM_KEY, key);
+        model.addAttribute(PARAM_TOKEN, token);
+        return "deviceRecord/list";
+    }
+
+    @ResponseBody
+    @RequestMapping(value = "/list", method = RequestMethod.POST)
+    public Result<Page<DeviceRecord>> list(@RequestBody(required = false) DeviceRecordQuery params,
+                                           @RequestParam(required = false) String key,
+                                           @RequestParam(required = false) String token) throws Exception {
+        if (StringUtils.isBlank(key) || StringUtils.isBlank(token)) {
+            throw new IllegalArgumentException("[Param] key or token must be not null.");
+        }
+
+        if (params == null) {
+            params = new DeviceRecordQuery();
+        }
+
+        return deviceRecordService.getDeviceRecordList(params);
+    }
+
+}

+ 65 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/ExamController.java

@@ -0,0 +1,65 @@
+package cn.com.qmth.examcloud.app.controller;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+import javax.validation.Valid;
+
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.ModelAttribute;
+import org.springframework.web.bind.annotation.PathVariable;
+import org.springframework.web.bind.annotation.PostMapping;
+import org.springframework.web.bind.annotation.RequestHeader;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import cn.com.qmth.examcloud.app.model.GetYunSignatureReq;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreOeService;
+import cn.com.qmth.examcloud.web.support.ControllerSupport;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+
+@RestController
+@RequestMapping("${$rmp}/v2/exam")
+@Api(tags = "考试相关接口")
+public class ExamController extends ControllerSupport {
+
+    @Autowired
+    private CoreOeService oeService;
+
+	@ApiOperation(value = "查询考生的考试批次属性集")
+	@GetMapping("getExamProperty/{examId}/{keys}")
+	public Result getExamPropertyFromCacheByStudentSession(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token,@PathVariable Long examId,
+			@PathVariable String keys) throws Exception{
+	    return oeService.getExamPropertyFromCacheByStudentSession(key, token, examId, keys);
+	}
+	/**
+     * 获取云存储上传签名(微信小程序调用)
+     */
+    @ApiOperation(value = "获取文件上传签名")
+    @PostMapping("/yunSignature")
+    public Result getYunSignature(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token,@ModelAttribute @Valid GetYunSignatureReq req) throws Exception{
+        return oeService.getYunSignature(key, token, req);
+    }
+    
+    @ApiOperation(value = "获取在线考试待考列表")
+    @GetMapping("/queryExamList")
+    public Result queryExamList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception{
+        return oeService.queryExamList(key, token);
+    }
+    
+    @ApiOperation(value = "获取考试记录信息")
+    @GetMapping("/getEndExamInfo/{examRecordDataId}")
+    public Result getEndExamInfo(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token,@PathVariable Long examRecordDataId) throws Exception{
+        return oeService.getEndExamInfo(key, token,examRecordDataId);
+    }
+    
+    @ApiOperation(value = "根据examStudentId获取客观分信息")
+    @GetMapping("/queryObjectiveScoreList/{examStudentId}")
+    public Result queryObjectiveScoreList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token,@PathVariable Long examStudentId) throws Exception{
+        return oeService.queryObjectiveScoreList(key, token,examStudentId);
+    }
+
+}

+ 29 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/IndexController.java

@@ -0,0 +1,29 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 15:00:21.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import org.springframework.stereotype.Controller;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RequestMethod;
+
+import javax.servlet.http.HttpServletRequest;
+
+/**
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Controller
+public class IndexController {
+
+    /*
+    @RequestMapping(value = "/", method = RequestMethod.GET)
+    public String index(HttpServletRequest request) throws Exception {
+        return "index";
+    }*/
+
+}

+ 102 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/OfflineExamController.java

@@ -0,0 +1,102 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-18 15:30:38.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreOeService;
+import cn.com.qmth.examcloud.app.service.CoreQuestionService;
+import cn.com.qmth.examcloud.app.service.UpYunService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Controller;
+import org.springframework.web.bind.annotation.*;
+import org.springframework.web.multipart.MultipartFile;
+import org.springframework.web.multipart.MultipartHttpServletRequest;
+
+import javax.servlet.http.HttpServletRequest;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+/**
+ * 离线考试相关接口
+ *
+ * @version v1.0
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Controller
+@RequestMapping("${$rmp}/v2/exam/offline")
+@Api(tags = "离线考试相关接口")
+public class OfflineExamController {
+    private static Logger log = LoggerFactory.getLogger(OfflineExamController.class);
+    @Autowired
+    private CoreOeService oeService;
+    @Autowired
+    private CoreQuestionService questionService;
+    @Autowired
+    private UpYunService upYunService;
+
+    @ResponseBody
+    @ApiOperation(value = "获取当前用户参加的离线课程列表接口")
+    @RequestMapping(value = "/course/list", method = {RequestMethod.POST})
+    public Result getOfflineExamCourseList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return oeService.getOfflineExamCourseList(key, token);
+    }
+
+    @ResponseBody
+    @ApiOperation(value = "离线考试的抽取考题接口")
+    @RequestMapping(value = "/record/start", method = {RequestMethod.POST})
+    public Result startOfflineExamRecord(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examStudentId) throws Exception {
+        return oeService.startOfflineExamRecord(key, token, examStudentId);
+    }
+
+    @ResponseBody
+    @ApiOperation(value = "上传作答文件接口")
+    @RequestMapping(value = "/paper/answer/upload", method = {RequestMethod.POST})
+    public Result uploadPaperAnswer(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examRecordId,
+                                    @RequestParam(required = false) String md5, HttpServletRequest request) throws Exception {
+        MultipartFile multipart = null;
+        try {
+            multipart = ((MultipartHttpServletRequest) request).getFile("file");
+        } catch (Exception e) {
+            log.error(e.getMessage());
+        }
+        if (multipart == null) {
+            return new Result().error("请选择要上传文件!");
+        }
+        return oeService.uploadPaperAnswer(key, token, examRecordId, multipart.getBytes(), multipart.getOriginalFilename(), md5);
+    }
+
+    @ResponseBody
+    @ApiOperation(value = "获取某份试卷的详细信息接口")
+    @RequestMapping(value = "/paper/detail", method = {RequestMethod.POST})
+    public Result getPaperDetail(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String paperId) throws Exception {
+        return questionService.getPaperDetail(key, token, paperId);
+    }
+
+    @ApiOperation(value = "下载考题接口")
+    @RequestMapping(value = "/paper/download", method = {RequestMethod.GET})
+    public String downloadPaper(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String paperId, @RequestParam String orgName) throws Exception {
+        String requestUrl = "redirectPermanent:" + questionService.downloadPaper(key, token, paperId, orgName);
+        log.debug(requestUrl);
+        return requestUrl;
+    }
+
+    @ApiOperation(value = "下载已上传的“作答文件”接口")
+    @RequestMapping(value = "/paper/answer/download", method = {RequestMethod.GET})
+    public String downloadPaperAnswer(@RequestParam String filePath) throws Exception {
+        String requestUrl = "redirect:" + upYunService.downloadPaperAnswer(filePath);
+        log.debug(requestUrl);
+        return requestUrl;
+    }
+
+}

+ 35 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/PaperController.java

@@ -0,0 +1,35 @@
+package cn.com.qmth.examcloud.app.controller;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.PathVariable;
+import org.springframework.web.bind.annotation.RequestHeader;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreOeService;
+import cn.com.qmth.examcloud.web.support.ControllerSupport;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+
+@RestController
+@RequestMapping("${$rmp}/v2/paper")
+@Api(tags = "试卷相关接口")
+public class PaperController extends ControllerSupport {
+
+    @Autowired
+    private CoreOeService oeService;
+
+	@ApiOperation(value = "根据Id获取试卷")
+	@GetMapping("{paperId}")
+	public Result getPaperById(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token,
+			@PathVariable String paperId) throws Exception{
+	    return oeService.getPaperById(key, token,paperId);
+	}
+
+
+}

+ 178 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/PracticeExamRestController.java

@@ -0,0 +1,178 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 17:53:55.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreExamWorkService;
+import cn.com.qmth.examcloud.app.service.CoreOeService;
+import cn.com.qmth.examcloud.app.service.CoreQuestionService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+/**
+ * 在线练习相关接口
+ *
+ * @version v1.0
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@RestController
+@RequestMapping("${$rmp}/v2")
+@Api(tags = "在线练习相关接口")
+public class PracticeExamRestController {
+    @Autowired
+    private CoreExamWorkService examWorkService;
+    @Autowired
+    private CoreOeService oeService;
+    @Autowired
+    private CoreQuestionService questionService;
+
+    @ApiOperation(value = "获取某考生的“考试批次”列表接口")
+    @RequestMapping(value = "/exam/practice/list", method = {RequestMethod.POST})
+    public Result getPracticeExamList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String studentId) throws Exception {
+        return examWorkService.getPracticeExamList(key, token, studentId);
+    }
+
+    @ApiOperation(value = "获取某考试批次下的课程列表接口")
+    @RequestMapping(value = "/exam/practice/course/list", method = {RequestMethod.POST})
+    public Result getPracticeExamCourseList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examId) throws Exception {
+        return oeService.getPracticeExamCourseList(key, token, examId);
+    }
+
+    @ApiOperation(value = "获取当前练习的剩余作答时间接口")
+    @RequestMapping(value = "/exam/record/heartbeat", method = {RequestMethod.POST})
+    public Result getExamRecordHeartbeat(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return oeService.getExamRecordHeartbeat(key, token);
+    }
+
+    @ApiOperation(value = "考生“开始练习”接口")
+    @RequestMapping(value = "/exam/record/start", method = {RequestMethod.POST})
+    public Result startPracticeExamRecord(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examStudentId) throws Exception {
+        return oeService.startPracticeExamRecord(key, token, examStudentId);
+    }
+
+    @ApiOperation(value = "获取结束练习的试卷大题结构信息接口")
+    @RequestMapping(value = "/exam/admin/record/paper/struct/list", method = {RequestMethod.POST})
+    public Result getExamAdminRecordPaperStructList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examRecordId,@RequestParam(required = false) String fromCache) throws Exception {
+        return oeService.getAdminExamPaperStructList(key, token, examRecordId,fromCache);
+    }
+    @ApiOperation(value = "获取当前练习的试卷大题结构信息接口")
+    @RequestMapping(value = "/exam/student/record/paper/struct/list", method = {RequestMethod.POST})
+    public Result getExamStudentRecordPaperStructList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examRecordId) throws Exception {
+        return oeService.getStudentExamPaperStructList(key, token, examRecordId);
+    }
+
+    @ApiOperation(value = "获取当前练习的考试基本信息接口")
+    @RequestMapping(value = "/exam/info/{examId}", method = {RequestMethod.POST})
+    public Result getExamInfo(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @PathVariable Long examId) throws Exception {
+        return examWorkService.getExamInfo(key, token, examId);
+    }
+
+    @ApiOperation(value = "获取当前练习的试卷试题列表接口")
+    @RequestMapping(value = "/exam/record/paper/question/list", method = {RequestMethod.POST})
+    public Result getExamRecordPaperQuestionList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return oeService.getExamRecordPaperQuestionList(key, token);
+    }
+
+    @ApiOperation(value = "获取考生作答的某个试题的详细信息接口")
+    @RequestMapping(value = "/exam/record/paper/question/detail/{questionId}/{examRecordId}", method = {RequestMethod.POST})
+    public Result questionDetail(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @PathVariable String questionId, @PathVariable String examRecordId) throws Exception {
+        return oeService.getExamRecordPaperQuestionDetail(key, token, questionId, examRecordId);
+    }
+
+    @ApiOperation(value = "保存或更新考生作答的某个试题答案接口")
+    @RequestMapping(value = "/exam/record/paper/question/answer/update", method = {RequestMethod.POST})
+    public Result updateExamRecordPaperQuestionAnswer(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam Integer order, @RequestParam String studentAnswer) throws Exception {
+        return oeService.updateExamRecordPaperQuestionAnswer(key, token, order, studentAnswer);
+    }
+
+    @ApiOperation(value = "当前练习的交卷接口")
+    @RequestMapping(value = "/exam/record/submit", method = {RequestMethod.POST})
+    public Result submitPracticeExamRecord(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return oeService.submitPracticeExamRecord(key, token);
+    }
+
+    @ApiOperation(value = "检查考生当前是否有正在进行的练习记录接口")
+    @RequestMapping(value = "/exam/record/online/check", method = {RequestMethod.POST})
+    public Result checkOnlineExamRecord(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return oeService.checkOnlineExamRecord(key, token);
+    }
+
+    @ApiOperation(value = "获取当前考生的当前课程的历史练习记录接口")
+    @RequestMapping(value = "/exam/record/practice/course/history/list", method = {RequestMethod.POST})
+    public Result getExamRecordPracticeHistoryList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examStudentId) throws Exception {
+        return oeService.getExamRecordPracticeHistoryList(key, token, examStudentId);
+    }
+
+    @ApiOperation(value = "获取成绩报告的答题情况统计接口")
+    @RequestMapping(value = "/exam/record/total", method = {RequestMethod.POST})
+    public Result getExamRecordTotalInfo(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examRecordId,@RequestParam(required = false) String fromCache) throws Exception {
+        return oeService.getExamRecordTotalInfo(key, token, examRecordId,fromCache);
+    }
+
+    @ApiOperation(value = "获取作答的题列表接口")
+    @RequestMapping(value = "/exam/record/paper/question/detail/list", method = {RequestMethod.POST})
+    public Result getExamRecordQuestionDetailList(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String examRecordId) throws Exception {
+        return oeService.getExamRecordQuestionDetailList(key, token, examRecordId);
+    }
+
+    @ApiOperation(value = "获取当前试题的音频已播放次数接口")
+    @RequestMapping(value = "/exam/record/paper/question/get/playtimes", method = {RequestMethod.POST})
+    public Result getExamRecordQuestionAudioPlayTimes(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String questionId) throws Exception {
+        return oeService.getExamRecordQuestionAudioPlayTimes(key, token, questionId);
+    }
+
+    @ApiOperation(value = "更新当前试题的音频已播放次数接口")
+    @RequestMapping(value = "/exam/record/paper/question/update/playtimes", method = {RequestMethod.POST})
+    public Result updateExamRecordQuestionAudioPlayTimes(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String questionId, @RequestParam String mediaName) throws Exception {
+        return oeService.updateExamRecordQuestionAudioPlayTimes(key, token, questionId, mediaName);
+    }
+
+    @ApiOperation(value = "开考前查询“考试说明”")
+    @RequestMapping(value = "/exam/practice/before/{examId}/{type}", method = {RequestMethod.POST})
+    public Result getBeforeExamRemark(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @PathVariable Long examId, @PathVariable String type) throws Exception {
+        return examWorkService.getBeforeExamRemark(key, token, examId, type);
+    }
+
+    @ApiOperation(value = "查询练习记录配置信息")
+    @RequestMapping(value = "/exam/practice/end/findExamRecordDataEntity/{examRecordDataId}", method = {RequestMethod.POST})
+    public Result findExamRecordDataEntity(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @PathVariable Long examRecordDataId,@RequestParam(required = false) String fromCache) throws Exception {
+        return oeService.findExamRecordDataEntity(key, token, examRecordDataId, fromCache);
+    }
+
+    @ApiOperation(value = "查询练习记录试题列表")
+    @RequestMapping(value = "/exam/practice/end/getExamRecordQuestions/{examRecordDataId}", method = {RequestMethod.POST})
+    public Result getExamRecordQuestions(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @PathVariable Long examRecordDataId,@RequestParam(required = false) String fromCache) throws Exception {
+        return oeService.getExamRecordQuestions(key, token, examRecordDataId,fromCache);
+    }
+
+    @ApiOperation(value = "查询某个试题内容")
+    @RequestMapping(value = "/exam/practice/end/question", method = {RequestMethod.POST})
+    public Result getQuestion(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String courseCode,
+                              @RequestParam Long examId, @RequestParam String groupCode, @RequestParam String questionId) throws Exception {
+        return questionService.getQuestion(key, token, courseCode, examId, groupCode, questionId);
+    }
+
+    @ApiOperation(value = "获取冻结时间")
+    @RequestMapping(value = "/exam/practice/getFreezeTime", method = {RequestMethod.POST})
+    public Result getFreezeTime(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam Long examId) throws Exception {
+        return examWorkService.getFreezeTime(key, token, examId);
+    }
+
+    @ApiOperation(value = "获取文件上传类型")
+    @RequestMapping(value = "/exam/practice/getUpLoadType/{examId}", method = {RequestMethod.POST})
+    public Result getUpLoadType(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @PathVariable Long examId) throws Exception {
+        return examWorkService.getUpLoadType(key, token, examId);
+    }
+
+}

+ 48 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/RouterController.java

@@ -0,0 +1,48 @@
+/*
+ * *************************************************
+ * Copyright (c) 2019 QMTH. All Rights Reserved.
+ * Created by Deason on 2019-08-19 10:38:43.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import cn.com.qmth.examcloud.app.core.router.Router;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.RouterService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+/**
+ * 路由相关接口
+ *
+ * @author: fengdesheng
+ * @since: 2019/8/19
+ */
+@RestController
+@RequestMapping("${$rmp}/")
+@Api(tags = "路由相关接口")
+public class RouterController {
+    private static Logger log = LoggerFactory.getLogger(RouterController.class);
+
+    @Autowired
+    private RouterService routerService;
+
+    @ApiOperation(value = "路由接口")
+    @PostMapping(value = "/router")
+    public Result router(@RequestHeader(name = PARAM_APP_KEY) String key,
+                         @RequestHeader(name = PARAM_APP_TOKEN) String token,
+                         @RequestBody Router router) {
+        router.setKey(key);
+        router.setToken(token);
+        return routerService.execute(router);
+    }
+
+}

+ 65 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/SystemRestController.java

@@ -0,0 +1,65 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-18 10:38:43.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreAuthService;
+import cn.com.qmth.examcloud.app.service.CoreOeService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+/**
+ * 系统服务相关接口
+ *
+ * @version v1.0
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@RestController
+@RequestMapping("${$rmp}/v2")
+@Api(tags = "系统服务相关接口")
+public class SystemRestController {
+    private static Logger log = LoggerFactory.getLogger(SystemRestController.class);
+    @Autowired
+    private CoreOeService oeService;
+    @Autowired
+    private CoreAuthService authService;
+
+    @ApiOperation(value = "获取服务器当前时间接口")
+    @RequestMapping(value = "/system/currentTime", method = {RequestMethod.POST})
+    public Result getCurrentTime(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return oeService.getCurrentTime(key, token);
+    }
+
+    @ApiOperation(value = "获取Token接口")
+    @RequestMapping(value = "/token", method = {RequestMethod.POST})
+    public Result getToken(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        log.debug("key:" + key);
+        return new Result().success(token);
+    }
+
+    @ApiOperation(value = "获取短信验证码接口")
+    @RequestMapping(value = "/send/sms/code", method = {RequestMethod.POST})
+    public Result sendSmsCode(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String phone) throws Exception {
+        return authService.sendSmsCode(key, token, phone);
+    }
+
+    @ApiOperation(value = "获取短信验证码接口(不带token)")
+    @RequestMapping(value = "/send/sms/code4Student", method = {RequestMethod.POST})
+    public Result sendSmsCode(@RequestParam String phone) throws Exception {
+        return authService.code4Student(phone, true);
+    }
+
+}

+ 129 - 0
src/main/java/cn/com/qmth/examcloud/app/controller/UserAuthRestController.java

@@ -0,0 +1,129 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 17:50:31.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.controller;
+
+import cn.com.qmth.examcloud.app.model.LoginInfo;
+import cn.com.qmth.examcloud.app.model.LoginType;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.model.UserInfo;
+import cn.com.qmth.examcloud.app.service.CoreAuthService;
+import cn.com.qmth.examcloud.app.service.CoreBasicService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_KEY;
+import static cn.com.qmth.examcloud.app.model.Constants.PARAM_APP_TOKEN;
+
+/**
+ * 认证中心相关接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@RestController
+@RequestMapping("${$rmp}/v2")
+@Api(tags = "认证中心相关接口")
+public class UserAuthRestController {
+    private final static Logger log = LoggerFactory.getLogger(UserAuthRestController.class);
+    @Autowired
+    private CoreAuthService authService;
+    @Autowired
+    private CoreBasicService basicService;
+
+    @ApiOperation(value = "登录接口", notes = "参数accountType值:STUDENT_IDENTITY_NUMBER、STUDENT_CODE、STUDENT_PHONE")
+    @RequestMapping(value = "/user/login", method = {RequestMethod.POST})
+    public Result<UserInfo> login(@RequestParam String account,
+                                  @RequestParam String password,
+                                  @RequestParam String accountType,
+                                  @RequestParam(required = false) Long rootOrgId,
+                                  @RequestParam(required = false) String domain,
+                                  @RequestHeader String deviceId) throws Exception {
+        LoginInfo loginInfo = new LoginInfo(account, password, accountType, rootOrgId, domain, deviceId, null);
+        Result<UserInfo> result = authService.login(loginInfo);
+
+        if (result.isSuccess() && result.getData() != null) {
+            //登录成功后缓存Token信息
+            UserInfo userInfo = result.getData();
+            loginInfo.setUserId(userInfo.getUserId());
+            loginInfo.setRootOrgId(userInfo.getRootOrgId());
+            loginInfo.setUserName(userInfo.getDisplayName());
+            loginInfo.setKey(userInfo.getKey());
+            loginInfo.setToken(userInfo.getToken());
+            loginInfo.setAppToken(userInfo.getToken());
+
+            authService.cacheLoginInfo(loginInfo, userInfo.getKey());
+            log.info(String.format("key:%s token:%s", userInfo.getKey(), userInfo.getToken()));
+        }
+
+        return result;
+    }
+
+    @ApiOperation(value = "验证码登录接口")
+    @RequestMapping(value = "/user/login/verify", method = {RequestMethod.POST})
+    public Result<UserInfo> verifyLogin(@RequestParam String account,
+                                        @RequestParam String smsCode,
+                                        @RequestParam(required = false) Long rootOrgId,
+                                        @RequestParam(required = false) String domain,
+                                        @RequestHeader String deviceId) throws Exception {
+        LoginInfo loginInfo = new LoginInfo(account, null, LoginType.STUDENT_PHONE.name(), rootOrgId, domain, deviceId, smsCode);
+        Result<UserInfo> result = authService.login(loginInfo);
+
+        if (result.isSuccess() && result.getData() != null) {
+            //登录成功后缓存Token信息
+            UserInfo userInfo = result.getData();
+            loginInfo.setUserId(userInfo.getUserId());
+            loginInfo.setRootOrgId(userInfo.getRootOrgId());
+            loginInfo.setUserName(userInfo.getDisplayName());
+            loginInfo.setToken(userInfo.getToken());
+            loginInfo.setKey(userInfo.getKey());
+            loginInfo.setAppToken(userInfo.getToken());
+
+            authService.cacheLoginInfo(loginInfo, userInfo.getKey());
+            log.info(String.format("key:%s token:%s", userInfo.getKey(), userInfo.getToken()));
+        }
+
+        return result;
+    }
+
+    @ApiOperation(value = "登出接口")
+    @RequestMapping(value = "/user/logout", method = {RequestMethod.POST})
+    public Result logout(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        return authService.logout(key, token);
+    }
+
+    @ApiOperation(value = "获取用户信息接口")
+    @RequestMapping(value = "/user/info", method = {RequestMethod.POST})
+    public Result getUserInfo(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token) throws Exception {
+        //return authService.getUserInfo(key, token);
+        return basicService.getStudentInfo(key, token);
+    }
+
+    @ApiOperation(value = "修改密码接口")
+    @RequestMapping(value = "/user/update/password", method = {RequestMethod.POST})
+    public Result updatePassword(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam Long studentId, @RequestParam String password,
+                                 @RequestParam String newPassword) throws Exception {
+        return authService.updateStudentPassword(key, token, studentId, password, newPassword);
+    }
+
+    @ApiOperation(value = "重置密码接口")
+    @RequestMapping(value = "/user/reset/password", method = {RequestMethod.POST})
+    public Result updateNewPassword(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String newPassword) throws Exception {
+        return authService.resetStudentPassword(key, token, newPassword);
+    }
+
+    @ApiOperation(value = "保存用户绑定的手机号接口")
+    @RequestMapping(value = "/user/binding/phone", method = {RequestMethod.POST})
+    public Result userBindingPhone(@RequestHeader(name = PARAM_APP_KEY) String key, @RequestHeader(name = PARAM_APP_TOKEN) String token, @RequestParam String phone, @RequestParam String code) throws Exception {
+        return authService.userBindingPhone(key, token, phone, code);
+    }
+
+}

+ 74 - 0
src/main/java/cn/com/qmth/examcloud/app/core/CloudDiscoveryClient.java

@@ -0,0 +1,74 @@
+/*
+ * *************************************************
+ * Copyright (c) 2019 QMTH. All Rights Reserved.
+ * Created by Deason on 2019-05-29 11:54:21.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core;
+
+import cn.com.qmth.examcloud.commons.exception.ExamCloudRuntimeException;
+import com.google.common.cache.Cache;
+import com.google.common.cache.CacheBuilder;
+import com.google.common.collect.Lists;
+import org.apache.commons.collections.CollectionUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cloud.client.ServiceInstance;
+import org.springframework.cloud.client.discovery.DiscoveryClient;
+import org.springframework.stereotype.Component;
+
+import java.util.List;
+import java.util.Random;
+import java.util.concurrent.TimeUnit;
+
+/**
+ * 系统实例
+ *
+ * @author: fengdesheng
+ * @since: 2019/5/29
+ */
+@Component
+public class CloudDiscoveryClient {
+
+    @Autowired(required = false)
+    private DiscoveryClient discoveryClient;
+
+    private Cache<String, List<String>> cache = CacheBuilder.newBuilder()
+            .maximumSize(1000)//最大缓存对象数
+            .expireAfterWrite(60, TimeUnit.SECONDS)//过期时间
+            .concurrencyLevel(8)//并发级别,即允许最多N个线程并发更新, 默认值为4
+            .recordStats()//开启记录状态
+            .build();
+
+    /**
+     * 获取实例的URL
+     */
+    public String getInstanceUrl(String appName) {
+        List<String> appUrls = cache.getIfPresent(appName);
+        if (CollectionUtils.isNotEmpty(appUrls)) {
+            //随机分配一个地址
+            int n = this.randomInt(0, appUrls.size() - 1);
+            return appUrls.get(n);
+        }
+
+        List<ServiceInstance> instances = discoveryClient.getInstances(appName);
+        if (CollectionUtils.isEmpty(instances)) {
+            throw new ExamCloudRuntimeException("No Instance " + appName);
+        }
+
+        appUrls = Lists.newArrayList();
+        for (ServiceInstance instance : instances) {
+            appUrls.add(String.format("http://%s:%s", instance.getHost(), instance.getPort()));
+        }
+        cache.put(appName, appUrls);
+
+        //随机分配一个地址
+        int n = this.randomInt(0, appUrls.size() - 1);
+        return appUrls.get(n);
+    }
+
+    public int randomInt(int min, int max) {
+        return new Random().nextInt(max - min + 1) + min;
+    }
+
+}

+ 43 - 0
src/main/java/cn/com/qmth/examcloud/app/core/Constants.java

@@ -0,0 +1,43 @@
+/*
+ * *************************************************
+ * Copyright (c) 2019 QMTH. All Rights Reserved.
+ * Created by Deason on 2019-05-29 11:33:47.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core;
+
+/**
+ * 系统常量
+ *
+ * @author: fengdesheng
+ * @since: 2019/5/29
+ */
+public interface Constants {
+
+    /**
+     * 基础信息平台
+     */
+    String EC_CORE_BASIC = "EC-CORE-BASIC";
+
+    /**
+     * 考务平台
+     */
+    String EC_CORE_EXAMWORK = "EC-CORE-EXAMWORK";
+
+    /**
+     * 网考管理端平台
+     */
+    String EC_CORE_OE_ADMIN = "EC-CORE-OE-ADMIN";
+
+    /**
+     * 网考学生端平台
+     */
+    String EC_CORE_OE_STUDENT = "EC-CORE-OE-STUDENT";
+
+    /**
+     * 题库平台
+     */
+    String EC_CORE_QUESTION = "EC-CORE-QUESTION";
+
+}

+ 64 - 0
src/main/java/cn/com/qmth/examcloud/app/core/SysProperty.java

@@ -0,0 +1,64 @@
+/*
+ * *************************************************
+ * Copyright (c) 2019 QMTH. All Rights Reserved.
+ * Created by Deason on 2019-05-29 11:31:01.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core;
+
+import cn.com.qmth.examcloud.app.core.router.Server;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.stereotype.Component;
+
+/**
+ * 属性配置服务类
+ */
+@Component
+public class SysProperty {
+    private static Logger log = LoggerFactory.getLogger(SysProperty.class);
+    @Autowired
+    private CloudDiscoveryClient discoveryClient;
+
+    @Value("${$upyun.site.1.domain}")
+    private String domain;//又拍云文件服务
+
+    public String getProxyUrl(Server server) {
+        return discoveryClient.getInstanceUrl(server.getInstanceName());
+    }
+
+    public String getCoreBasicUrl() {
+        return discoveryClient.getInstanceUrl(Constants.EC_CORE_BASIC);
+    }
+
+    public String getCoreExamWorkUrl() {
+        return discoveryClient.getInstanceUrl(Constants.EC_CORE_EXAMWORK);
+    }
+
+    public String getCoreOeStudentUrl() {
+        return discoveryClient.getInstanceUrl(Constants.EC_CORE_OE_STUDENT);
+    }
+
+    public String getCoreOeAdminUrl() {
+        return discoveryClient.getInstanceUrl(Constants.EC_CORE_OE_ADMIN);
+    }
+
+    public String getCoreQuestionUrl() {
+        return discoveryClient.getInstanceUrl(Constants.EC_CORE_QUESTION);
+    }
+
+    public String getCoreAuthUrl() {
+        return discoveryClient.getInstanceUrl(Constants.EC_CORE_BASIC);
+    }
+
+    public String getDomain() {
+        if (domain != null) {
+            return domain.trim();
+        }
+        return "";
+    }
+
+}

+ 127 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/AccessInterceptor.java

@@ -0,0 +1,127 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 14:15:02.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import cn.com.qmth.examcloud.app.model.DeviceRecord;
+import cn.com.qmth.examcloud.app.service.DeviceRecordService;
+import cn.com.qmth.examcloud.commons.util.ThreadLocalUtil;
+import org.apache.commons.lang3.StringUtils;
+import org.apache.logging.log4j.ThreadContext;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.util.Date;
+import java.util.Map;
+
+import static cn.com.qmth.examcloud.app.model.Constants.*;
+
+/**
+ * API请求拦截器
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public class AccessInterceptor extends HandlerInterceptorAdapter {
+    private final static Logger log = LoggerFactory.getLogger(AccessInterceptor.class);
+    @Autowired
+    private DeviceRecordService deviceRecordService;
+
+    @Override
+    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
+        String mapping = request.getServletPath().toLowerCase();
+        if (mapping.endsWith(".js") || mapping.endsWith(".css")
+                || mapping.endsWith(".jpg") || mapping.endsWith(".png")
+                || mapping.endsWith(".gif") || mapping.endsWith(".map")
+                || mapping.endsWith(".woff") || mapping.endsWith(".ico")) {
+            return true;
+        }
+
+        // 设置log4j线程上下文
+        String traceId = ThreadLocalUtil.next();
+        ThreadContext.put("TRACE_ID", traceId);
+
+        if (request.getServletPath().contains("/router")) {
+            return true;
+        }
+
+        //异步保存设备访问记录
+        DeviceRecord record = this.parseParams(request);
+        deviceRecordService.addDeviceRecord(record);
+        return true;
+    }
+
+    private DeviceRecord parseParams(HttpServletRequest request) {
+        String params = this.getParams(request);
+        String url = request.getServletPath() + params;
+        String ip = this.getIp(request);
+        //封装设备访问记录信息
+        DeviceRecord record = new DeviceRecord();
+        record.setSystem(request.getHeader("system"));
+        record.setDeviceId(request.getHeader("deviceId"));
+        record.setNetType(request.getHeader("netType"));
+        record.setBrand(request.getHeader("brand"));
+        record.setModel(request.getHeader("model"));
+        record.setSysVersion(request.getHeader("sysVersion"));
+        record.setAppVersion(request.getHeader("appVersion"));
+        record.setAppCode(request.getHeader("appCode"));
+        record.setPatchCode(request.getHeader("patchCode"));
+        record.setAccount(request.getHeader("account"));
+        record.setLoginKey(request.getHeader(PARAM_APP_KEY));
+        record.setLoginToken(request.getHeader(PARAM_APP_TOKEN));
+        record.setUrl(url);
+        record.setIp(ip);
+        record.setCreateDate(new Date());
+        return record;
+    }
+
+    private String getParams(HttpServletRequest request) {
+        StringBuilder params = new StringBuilder();
+        Map<String, String[]> map = request.getParameterMap();
+        for (Map.Entry<String, String[]> entry : map.entrySet()) {
+            String name = entry.getKey();
+            String[] values = entry.getValue();
+            params.append("&").append(name).append("=");
+            if (values.length > 0) {
+                if (name.toLowerCase().indexOf(PARAM_PSW) >= 0) {
+                    params.append("***");
+                } else {
+                    params.append(values[0]);
+                }
+            }
+        }
+        return params.toString().replaceFirst("&", "?");
+    }
+
+    private String getIp(HttpServletRequest request) {
+        String ip = request.getHeader("X-Forwarded-For");
+        if (StringUtils.isEmpty(ip) || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("Proxy-Client-IP");
+        }
+        if (StringUtils.isEmpty(ip) || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("WL-Proxy-Client-IP");
+        }
+        if (StringUtils.isEmpty(ip) || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("HTTP_CLIENT_IP");
+        }
+        if (StringUtils.isEmpty(ip) || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("HTTP_X_FORWARDED_FOR");
+        }
+        if (StringUtils.isEmpty(ip) || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getRemoteAddr();
+        }
+        if (StringUtils.isNotEmpty(ip)) {
+            ip = ip.replaceAll("0:0:0:0:0:0:0:1", "127.0.0.1");
+        }
+        return ip;
+    }
+
+}

+ 38 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/ControllerAdviceHandler.java

@@ -0,0 +1,38 @@
+package cn.com.qmth.examcloud.app.core.config;
+
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.commons.exception.StatusException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.web.bind.annotation.ControllerAdvice;
+import org.springframework.web.bind.annotation.ExceptionHandler;
+import org.springframework.web.bind.annotation.ResponseBody;
+
+@ResponseBody
+@ControllerAdvice
+public class ControllerAdviceHandler {
+
+    private final static Logger log = LoggerFactory.getLogger(ControllerAdviceHandler.class);
+
+    @ResponseBody
+    @ExceptionHandler(value = RuntimeException.class)
+    public Result handle(RuntimeException e) {
+        log.error(e.getMessage(), e);
+        return new Result().error(e.getMessage());
+    }
+
+    @ResponseBody
+    @ExceptionHandler(value = Exception.class)
+    public Result handle(Exception e) {
+        log.error(e.getMessage(), e);
+        return new Result().error(e.getMessage());
+    }
+
+    @ResponseBody
+    @ExceptionHandler(value = StatusException.class)
+    public Result handle(StatusException e) {
+        log.error(e.getMessage(), e);
+        return new Result().error(e.getMessage());
+    }
+
+}

+ 66 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/CustomHttpServletRequest.java

@@ -0,0 +1,66 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-01 14:43:30.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletRequestWrapper;
+import java.util.*;
+
+/**
+ * 自定义Request Header
+ *
+ * @author: fengdesheng
+ * @since: 2018/8/1
+ */
+public class CustomHttpServletRequest extends HttpServletRequestWrapper {
+    private final Set<String> names = new HashSet<>();
+    private final Map<String, String> headers = new HashMap<>();
+
+    public CustomHttpServletRequest(HttpServletRequest request) {
+        super(request);
+        Enumeration<String> enumeration = super.getHeaderNames();
+        while (enumeration.hasMoreElements()) {
+            names.add(enumeration.nextElement());
+        }
+    }
+
+    public void addHeader(String name, String value) {
+        if (name == null) {
+            return;
+        }
+        this.headers.put(name, value);
+        this.names.add(name);
+    }
+
+    @Override
+    public String getHeader(String name) {
+        if (headers.containsKey(name)) {
+            return headers.get(name);
+        }
+        return super.getHeader(name);
+    }
+
+    @Override
+    public Enumeration<String> getHeaders(String name) {
+        if (headers.containsKey(name)) {
+            String value = headers.get(name);
+            if (value != null) {
+                Set<String> set = new HashSet<>();
+                set.add(value);
+                return Collections.enumeration(set);
+            }
+        }
+        return super.getHeaders(name);
+    }
+
+    @Override
+    public Enumeration<String> getHeaderNames() {
+        return Collections.enumeration(names);
+    }
+
+}

+ 42 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/FilterConfig.java

@@ -0,0 +1,42 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-01 15:18:56.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import cn.com.qmth.examcloud.app.service.CoreAuthService;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
+import org.springframework.boot.web.servlet.FilterRegistrationBean;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * 请求过滤器
+ *
+ * @author: fengdesheng
+ * @since: 2018/8/1
+ */
+@Configuration
+public class FilterConfig {
+
+    @Bean
+    @ConditionalOnBean({CoreAuthService.class})
+    public FilterRegistrationBean filterRegistrationBean(CoreAuthService authService) {
+        FilterRegistrationBean registrationBean = new FilterRegistrationBean();
+        TokenFilter tokenFilter = new TokenFilter();
+        tokenFilter.setAuthService(authService);
+        registrationBean.setFilter(tokenFilter);
+        List<String> urlPatterns = new ArrayList<>();
+        //过滤器UrlPatterns通配符为一个"*"
+        urlPatterns.add("/api/*");
+        registrationBean.setUrlPatterns(urlPatterns);
+        return registrationBean;
+    }
+
+}

+ 36 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/InterceptorConfig.java

@@ -0,0 +1,36 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 14:15:02.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
+import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
+
+/**
+ * 请求拦截器
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Configuration
+public class InterceptorConfig implements WebMvcConfigurer {
+
+    @Bean
+    public AccessInterceptor accessInterceptor() {
+        return new AccessInterceptor();
+    }
+
+    @Override
+    public void addInterceptors(InterceptorRegistry registry) {
+        // 拦截器PathPatterns通配符为两个"*"
+        String[] excludes = new String[]{"/", "/error", "/webjars/**", "/doc.html", "/api/app-api/device/record/list"};
+        registry.addInterceptor(accessInterceptor()).addPathPatterns("/api/**").excludePathPatterns(excludes);
+    }
+
+}

+ 39 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/RedirectViewResolver.java

@@ -0,0 +1,39 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-21 17:51:26.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import org.springframework.http.HttpStatus;
+import org.springframework.stereotype.Component;
+import org.springframework.web.servlet.View;
+import org.springframework.web.servlet.ViewResolver;
+import org.springframework.web.servlet.view.RedirectView;
+
+import java.util.Locale;
+
+/**
+ * Resolver HTTP STATUS 301
+ *
+ * @author: fengdesheng
+ * @since: 2018/8/21
+ */
+@Component
+public class RedirectViewResolver implements ViewResolver {
+    public static final String REDIRECT_PERMANENT_PREFIX = "redirectPermanent:";
+
+    @Override
+    public View resolveViewName(String viewName, Locale locale) throws Exception {
+        if (viewName.startsWith(REDIRECT_PERMANENT_PREFIX)) {
+            String redirectUrl = viewName.substring(REDIRECT_PERMANENT_PREFIX.length());
+            RedirectView view = new RedirectView(redirectUrl);
+            view.setStatusCode(HttpStatus.MOVED_PERMANENTLY);
+            return view;
+        }
+        return null;
+    }
+
+}

+ 44 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/RedisConfig.java

@@ -0,0 +1,44 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-02 09:21:53.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import com.fasterxml.jackson.annotation.JsonAutoDetect;
+import com.fasterxml.jackson.annotation.PropertyAccessor;
+import com.fasterxml.jackson.databind.DeserializationFeature;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.data.redis.connection.RedisConnectionFactory;
+import org.springframework.data.redis.core.RedisTemplate;
+import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
+import org.springframework.data.redis.serializer.StringRedisSerializer;
+
+/**
+ * @author: fengdesheng
+ * @since: 2018/8/2
+ */
+@Configuration
+public class RedisConfig {
+
+    @Bean
+    public RedisTemplate<String, Object> strRedisTemplate(RedisConnectionFactory redisConnectionFactory) {
+        RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
+        Jackson2JsonRedisSerializer<?> jacksonSerializer = new Jackson2JsonRedisSerializer<>(Object.class);
+        ObjectMapper objectMapper = new ObjectMapper();
+        objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
+        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
+        objectMapper.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
+        jacksonSerializer.setObjectMapper(objectMapper);
+        redisTemplate.setValueSerializer(jacksonSerializer);
+        redisTemplate.setKeySerializer(new StringRedisSerializer());
+        redisTemplate.setConnectionFactory(redisConnectionFactory);
+        redisTemplate.afterPropertiesSet();
+        return redisTemplate;
+    }
+
+}

+ 44 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/SwaggerConfig.java

@@ -0,0 +1,44 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 14:15:16.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import io.swagger.annotations.ApiOperation;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import springfox.documentation.builders.ApiInfoBuilder;
+import springfox.documentation.builders.PathSelectors;
+import springfox.documentation.builders.RequestHandlerSelectors;
+import springfox.documentation.service.ApiInfo;
+import springfox.documentation.spi.DocumentationType;
+import springfox.documentation.spring.web.plugins.Docket;
+import springfox.documentation.swagger2.annotations.EnableSwagger2;
+
+@Configuration
+@EnableSwagger2
+public class SwaggerConfig {
+
+    @Bean
+    public Docket buildDocket() {
+        return new Docket(DocumentationType.SWAGGER_2)
+                .groupName("Version 1.0")
+                .apiInfo(buildApiInfo())
+                .useDefaultResponseMessages(false)
+                .select()
+                .apis(RequestHandlerSelectors.withMethodAnnotation(ApiOperation.class))
+                .paths(PathSelectors.any())
+                .build();
+    }
+
+    public ApiInfo buildApiInfo() {
+        return new ApiInfoBuilder()
+                .title("考生端APP接口文档")
+                .version("1.0")
+                .build();
+    }
+
+}

+ 255 - 0
src/main/java/cn/com/qmth/examcloud/app/core/config/TokenFilter.java

@@ -0,0 +1,255 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-01 15:17:10.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.config;
+
+import cn.com.qmth.examcloud.app.model.*;
+import cn.com.qmth.examcloud.app.service.CoreAuthService;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.http.HttpStatus;
+
+import javax.servlet.FilterConfig;
+import javax.servlet.*;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.io.IOException;
+import java.io.PrintWriter;
+import java.util.Date;
+
+import static cn.com.qmth.examcloud.app.model.Constants.PLATFORM_SESSION_EXPIRE_TIME;
+
+/**
+ * 认证信息过滤器
+ *
+ * @author: fengdesheng
+ * @since: 2018/8/1
+ */
+public class TokenFilter implements Filter {
+
+    private final static Logger log = LoggerFactory.getLogger(TokenFilter.class);
+
+    private CoreAuthService authService;
+
+    @Override
+    public void init(FilterConfig filterConfig) throws ServletException {
+        log.debug("init");
+    }
+
+    @Override
+    public void destroy() {
+        log.debug("destroy");
+    }
+
+    @Override
+    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
+            throws IOException, ServletException {
+        HttpServletRequest request = (HttpServletRequest) servletRequest;
+        HttpServletResponse response = (HttpServletResponse) servletResponse;
+
+        //移动端设备请求必须带“设备编号”参数
+        /*String deviceId = request.getHeader(Constants.PARAM_DEVICE_ID);
+        if (StringUtils.isBlank(deviceId)) {
+            Result result = new Result().error("[APP] deviceId must be not empty.");
+            this.render(response, result.toString(), true);
+            return;
+        }*/
+
+        Continue reqContinue = new Continue();
+        CustomHttpServletRequest customRequest = this.initCustomRequest(request, response, reqContinue);
+        if (customRequest != null) {
+            filterChain.doFilter(customRequest, servletResponse);
+            return;
+        }
+
+        if (reqContinue.yes) {
+            filterChain.doFilter(servletRequest, servletResponse);
+        }
+    }
+
+    private CustomHttpServletRequest initCustomRequest(HttpServletRequest request, HttpServletResponse response, Continue reqContinue) {
+        String url = request.getServletPath();
+        if (url.endsWith("/user/login") || url.endsWith("/user/login/verify")) {
+            //处理登录接口
+            String rootOrgIdStr = request.getParameter(Constants.PARAM_ROOT_ORG_ID);
+            String accountType = request.getParameter(Constants.PARAM_ACCOUNT_TYPE);
+            String account = request.getParameter(Constants.PARAM_ACCOUNT);
+
+            Long rootOrgId = null;
+            try {
+                rootOrgId = Long.parseLong(rootOrgIdStr);
+            } catch (NumberFormatException e) {
+                //ignore
+            }
+
+            if (rootOrgId == null) {
+                //rootOrgId = this.loadRootOrgId(request);
+                rootOrgId = authService.getRootOrgIdBySecurityPhone(account);
+            }
+
+            if (rootOrgId == null) {
+                reqContinue.yes = false;
+                this.renderError(response, new Result().error("学校ID获取失败!").toString());
+                return null;
+            }
+
+            boolean isOpen = authService.isOpenApp(rootOrgId);
+            log.info("[Check Open APP] Result is " + isOpen);
+            if (!isOpen) {
+                reqContinue.yes = false;
+                this.renderError(response, new Result().error("当前学校尚未开放APP功能!").toString());
+                return null;
+            }
+
+            boolean isDoing = authService.isDoingExam(rootOrgId, accountType, account);
+            log.info("[Check Doing Exam] Result is " + isDoing);
+            if (isDoing) {
+                reqContinue.yes = false;
+                this.renderError(response, new Result().error("尚在考试中不允许登录!").toString());
+                return null;
+            }
+        }
+
+        String appKey = request.getHeader(Constants.PARAM_APP_KEY);
+        String appToken = request.getHeader(Constants.PARAM_APP_TOKEN);
+        if (StringUtils.isBlank(appKey) || StringUtils.isBlank(appToken)) {
+            //key,token为空,则不用处理
+            return null;
+        }
+
+        //通过key获取原始登录信息
+        LoginInfo loginInfo = authService.getLoginInfo(appKey);
+        if (loginInfo == null) {
+            //原始登录信息为空,则代表尚未登录不用处理
+            return null;
+        }
+
+        //同一个账号不同移动端设备,支持互踢
+        if (!appToken.equals(loginInfo.getAppToken())) {
+            //App Token参数值无效
+            return null;
+        }
+
+        //处理已登录信息
+        CustomHttpServletRequest customRequest = new CustomHttpServletRequest(request);
+        boolean isAllow = this.filterAccessUrl(url);
+        if (!isAllow) {
+            if (loginInfo.hasExpired(PLATFORM_SESSION_EXPIRE_TIME)) {
+                if (StringUtils.isBlank(loginInfo.getPassword())) {
+                    log.info("Can't reLogin, no password.");
+                    return null;
+                }
+
+//                boolean isDoing = authService.isDoingExam(loginInfo.getRootOrgId(), loginInfo.getAccountType(), loginInfo.getAccount());
+//                log.info("[Check Doing Exam] result is " + isDoing);
+//                if (isDoing) {
+//                    reqContinue.yes = false;
+//                    this.renderError(response, new Result().error("尚在考试中不允许登录!").toString());
+//                    return null;
+//                }
+
+                //判断原始登录Token是否在有效时间内,否则自动登录续期
+                this.reLogin(loginInfo);
+            }
+        }
+
+        //将App Token替换为原始登录Token到request请求中
+        customRequest.addHeader(Constants.PARAM_APP_KEY, loginInfo.getKey());
+        customRequest.addHeader(Constants.PARAM_APP_TOKEN, loginInfo.getToken());
+
+        //初始化内部接口请求鉴权
+        authService.initRequestTrace();
+        return customRequest;
+    }
+
+    private Long loadRootOrgId(HttpServletRequest request) {
+        String accountType;
+        if (request.getServletPath().endsWith("/user/login/verify")) {
+            accountType = LoginType.STUDENT_PHONE.name();
+        } else {
+            accountType = request.getParameter("accountType");
+        }
+
+        String account = request.getParameter("account");
+        String password = request.getParameter("password");
+        String smsCode = request.getParameter("smsCode");
+        String domain = request.getParameter("domain");
+
+        LoginInfo loginInfo = new LoginInfo();
+        loginInfo.setNoSession(true);
+        loginInfo.setAccountType(accountType);
+        loginInfo.setAccount(account);
+        loginInfo.setPassword(password);
+        loginInfo.setSmsCode(smsCode);
+        loginInfo.setDomain(domain);
+
+        Result<UserInfo> result = authService.login(loginInfo);
+        if (result.isSuccess() && result.getData() != null) {
+            return result.getData().getRootOrgId();
+        }
+        return null;
+    }
+
+    private void reLogin(LoginInfo loginInfo) {
+        try {
+            Result<UserInfo> result = authService.login(loginInfo);
+            UserInfo userInfo = result.getData();
+            if (userInfo == null) {
+                log.debug("reLogin failed,Maybe user's password has changed.");
+                return;
+            }
+            //登录成功后缓存新的Token信息
+            log.info(String.format("key:%s newToken:%s", userInfo.getKey(), userInfo.getToken()));
+            loginInfo.setUserName(userInfo.getDisplayName());
+            loginInfo.setToken(userInfo.getToken());
+            loginInfo.setCreateTime(new Date());
+            authService.cacheLoginInfo(loginInfo, userInfo.getKey());
+        } catch (Exception e) {
+            log.error(e.getMessage());
+        }
+    }
+
+    private boolean filterAccessUrl(String url) {
+        //过滤不用处理的访问地址
+        if (url.contains("/user/logout")) {
+            return true;
+        }
+        return false;
+    }
+
+    private void renderError(HttpServletResponse response, String json) {
+        this.render(response, json, false);
+    }
+
+    private void render(HttpServletResponse response, String json, boolean isOk) {
+        PrintWriter out = null;
+        try {
+            response.setStatus(isOk ? HttpStatus.OK.value() : HttpStatus.INTERNAL_SERVER_ERROR.value());
+            response.setCharacterEncoding("UTF-8");
+            response.setContentType("application/json;charset=UTF-8");
+            out = response.getWriter();
+            out.print(json);
+        } catch (IOException e) {
+            log.error(e.getMessage(), e);
+        } finally {
+            if (out != null) {
+                out.close();
+            }
+        }
+    }
+
+    public void setAuthService(CoreAuthService authService) {
+        this.authService = authService;
+    }
+
+    class Continue {
+
+        boolean yes = true;
+    }
+
+}

+ 21 - 0
src/main/java/cn/com/qmth/examcloud/app/core/exception/ApiException.java

@@ -0,0 +1,21 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 15:21:38.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.exception;
+
+public class ApiException extends RuntimeException {
+    private static final long serialVersionUID = 0L;
+
+    public ApiException() {
+        super();
+    }
+
+    public ApiException(String message) {
+        super(message);
+    }
+
+}

+ 7 - 0
src/main/java/cn/com/qmth/examcloud/app/core/router/Method.java

@@ -0,0 +1,7 @@
+package cn.com.qmth.examcloud.app.core.router;
+
+public enum Method {
+
+    GET, POST, PUT, DELETE
+
+}

+ 98 - 0
src/main/java/cn/com/qmth/examcloud/app/core/router/Router.java

@@ -0,0 +1,98 @@
+package cn.com.qmth.examcloud.app.core.router;
+
+import io.swagger.annotations.ApiModelProperty;
+
+import java.io.Serializable;
+import java.util.Map;
+
+public class Router implements Serializable {
+
+    @ApiModelProperty(value = "服务名", example = "BASIC、EXAMWORK、OE_ADMIN、OE_STUDENT、QUESTION", required = true)
+    private Server server;
+
+    @ApiModelProperty(value = "URL地址", example = "/abc/1", required = true)
+    private String url;
+
+    @ApiModelProperty(value = "请求方式", example = "GET、POST、PUT、DELETE", required = true)
+    private Method method;
+
+    @ApiModelProperty(value = "请求Headers", example = "{'abc':'123'}", required = false)
+    private Map<String, String> headers;
+
+    @ApiModelProperty(value = "请求Params", example = "{'abc':'123'}", required = false)
+    private Map<String, String> params;
+
+    @ApiModelProperty(value = "请求Body", example = "JSON", required = false)
+    private String body;
+
+    @ApiModelProperty(value = "认证Key", required = false, hidden = true)
+    private String key;
+
+    @ApiModelProperty(value = "认证Token", required = false, hidden = true)
+    private String token;
+
+    public Server getServer() {
+        return server;
+    }
+
+    public void setServer(Server server) {
+        this.server = server;
+    }
+
+    public String getUrl() {
+        return url;
+    }
+
+    public void setUrl(String url) {
+        this.url = url;
+    }
+
+    public Method getMethod() {
+        return method;
+    }
+
+    public void setMethod(Method method) {
+        this.method = method;
+    }
+
+    public Map<String, String> getHeaders() {
+        return headers;
+    }
+
+    public void setHeaders(Map<String, String> headers) {
+        this.headers = headers;
+    }
+
+    public Map<String, String> getParams() {
+        return params;
+    }
+
+    public void setParams(Map<String, String> params) {
+        this.params = params;
+    }
+
+    public String getBody() {
+        return body;
+    }
+
+    public void setBody(String body) {
+        this.body = body;
+    }
+
+    public String getKey() {
+        return key;
+    }
+
+    public void setKey(String key) {
+        this.key = key;
+    }
+
+    public String getToken() {
+        return token;
+    }
+
+    public void setToken(String token) {
+        this.token = token;
+    }
+
+}

+ 40 - 0
src/main/java/cn/com/qmth/examcloud/app/core/router/Server.java

@@ -0,0 +1,40 @@
+package cn.com.qmth.examcloud.app.core.router;
+
+public enum Server {
+
+    /**
+     * 基础信息平台
+     */
+    BASIC("EC-CORE-BASIC"),
+
+    /**
+     * 考务平台
+     */
+    EXAMWORK("EC-CORE-EXAMWORK"),
+
+    /**
+     * 网考管理端平台
+     */
+    OE_ADMIN("EC-CORE-OE-ADMIN"),
+
+    /**
+     * 网考学生端平台
+     */
+    OE_STUDENT("EC-CORE-OE-STUDENT"),
+
+    /**
+     * 题库平台
+     */
+    QUESTION("EC-CORE-QUESTION");
+
+    private String instanceName;
+
+    Server(String instanceName) {
+        this.instanceName = instanceName;
+    }
+
+    public String getInstanceName() {
+        return instanceName;
+    }
+
+}

+ 41 - 0
src/main/java/cn/com/qmth/examcloud/app/core/utils/DateUtils.java

@@ -0,0 +1,41 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-18 14:25:01.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.utils;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.text.SimpleDateFormat;
+import java.util.Date;
+
+public class DateUtils {
+    private static Logger log = LoggerFactory.getLogger(DateUtils.class);
+
+    private static final String fmt = "yyyy-MM-dd HH:mm:ss";
+
+    public static String formatLongDate(String longStr) {
+        try {
+            Long value = Long.parseLong(longStr);
+            Date date = new Date(value);
+            return new SimpleDateFormat(fmt).format(date);
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+            return longStr;
+        }
+    }
+
+    public static String format(Date date) {
+        try {
+            return new SimpleDateFormat(fmt).format(date);
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+            return null;
+        }
+    }
+
+}

+ 46 - 0
src/main/java/cn/com/qmth/examcloud/app/core/utils/HttpClientBuilder.java

@@ -0,0 +1,46 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 16:38:13.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.utils;
+
+import okhttp3.OkHttpClient;
+
+/**
+ * HttpClient单例对象
+ *
+ * @author: fengdesheng
+ * @since: 2018/8/1
+ */
+public class HttpClientBuilder {
+    private static OkHttpClient client;
+
+    static {
+        client = Singleton.INSTANCE.getInstance();
+    }
+
+    public static OkHttpClient getClient() {
+        return client;
+    }
+
+    private enum Singleton {
+        /**
+         * Http Client Instance
+         */
+        INSTANCE;
+
+        private OkHttpClient instance;
+
+        Singleton() {
+            instance = new OkHttpClient();
+        }
+
+        public OkHttpClient getInstance() {
+            return instance;
+        }
+    }
+
+}

+ 169 - 0
src/main/java/cn/com/qmth/examcloud/app/core/utils/HttpUtils.java

@@ -0,0 +1,169 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-17 17:32:00.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.utils;
+
+import cn.com.qmth.examcloud.app.model.Constants;
+import cn.com.qmth.examcloud.app.model.ResBody;
+import cn.com.qmth.examcloud.app.model.Result;
+import com.alibaba.fastjson.JSON;
+import okhttp3.Request;
+import okhttp3.RequestBody;
+import okhttp3.Response;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.net.ConnectException;
+import java.util.UUID;
+
+import static cn.com.qmth.examcloud.app.model.Constants.*;
+
+/**
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public class HttpUtils {
+    private static Logger log = LoggerFactory.getLogger(HttpUtils.class);
+
+    public static String getRandomTraceId() {
+        String traceId = ThreadUtils.getTraceID();
+        if (StringUtils.isBlank(traceId)) {
+            traceId = UUID.randomUUID().toString().replace("-", "");
+        }
+        return traceId;
+    }
+
+    public static Result<String> doGet(String requestUrl, String key, String token) throws Exception {
+        String traceId = getRandomTraceId();
+
+        //封装请求参数
+        Request request = new Request.Builder()
+                .url(requestUrl)
+                .get()
+                .addHeader(PARAM_KEY, key)
+                .addHeader(PARAM_TOKEN, token)
+                .addHeader(PARAM_TRACE_ID, traceId)
+                .addHeader(PARAM_CLIENT, PARAM_CLIENT_VALUE)
+                .build();
+
+        //执行请求
+        log.info("[GET] " + requestUrl);
+        return call(request, traceId);
+    }
+
+    public static Result<String> doPost(String requestUrl, RequestBody formBody, String key, String token) throws Exception {
+        String traceId = getRandomTraceId();
+
+        //封装请求参数
+        Request request = new Request.Builder()
+                .url(requestUrl)
+                .post(formBody)
+                .addHeader(PARAM_KEY, key)
+                .addHeader(PARAM_TOKEN, token)
+                .addHeader(PARAM_TRACE_ID, traceId)
+                .addHeader(PARAM_CLIENT, PARAM_CLIENT_VALUE)
+                .build();
+
+        //执行请求
+        log.info("[POST] " + requestUrl);
+        return call(request, traceId);
+    }
+
+    public static Result<String> doPut(String requestUrl, RequestBody formBody, String key, String token) throws Exception {
+        String traceId = getRandomTraceId();
+
+        //封装请求参数
+        Request request = new Request.Builder()
+                .url(requestUrl)
+                .put(formBody)
+                .addHeader(PARAM_KEY, key)
+                .addHeader(PARAM_TOKEN, token)
+                .addHeader(PARAM_TRACE_ID, traceId)
+                .addHeader(PARAM_CLIENT, PARAM_CLIENT_VALUE)
+                .build();
+
+        //执行请求
+        log.info("[PUT] " + requestUrl);
+        return call(request, traceId);
+    }
+
+    public static Result<String> call(Request request, String traceId) throws Exception {
+        Response response;
+        try {
+            response = HttpClientBuilder.getClient().newCall(request).execute();
+        } catch (ConnectException e) {
+            log.error("[ConnectException] " + e.getMessage());
+            return new Result<>().error("服务访问失败!");
+        }
+
+        String bodyStr = response.body().string();
+        if (response.isSuccessful()) {
+            log.info("[Response] is success!");
+            return new Result().success(bodyStr);
+        } else {
+            log.warn("[Response] is fail! body:" + bodyStr);
+            ResBody body = new JsonMapper().fromJson(bodyStr, ResBody.class);
+            if (body != null && body.getCode() != null) {
+                if (Constants.CODE_403.equals(body.getCode())) {
+                    return new Result().noAuthError();
+                }
+                if (Constants.CODE_404.equals(body.getCode())) {
+                    return new Result().noFoundError();
+                }
+                if (Constants.CODE_0.equals(body.getCode())) {
+                    return new Result().error(body.getErrorMsg());
+                }
+                if (Constants.CODE_P001012.equals(body.getCode())) {
+                    return new Result().noAuthError();
+                }
+                if (Constants.CODE_B001012.equals(body.getCode())) {
+                    return new Result().noAuthError();
+                }
+                return new Result().error(body.getDesc());
+            }
+            return new Result().error(bodyStr);
+        }
+    }
+
+    /**
+     * 如果是JSON字符串,则过滤掉为空的属性
+     */
+    public static String filterNullAttributes(String str) {
+        if (StringUtils.isBlank(str)) {
+            return "";
+        }
+        try {
+            Object obj = JSON.parse(str);
+            return obj.toString();
+        } catch (Exception e) {
+            return str;
+        }
+    }
+
+    /**
+     * 替换字符串中所有的图片标签
+     */
+    public static String replaceImages(String html) {
+        if (html == null) {
+            return null;
+        }
+        String reg = "<\\s*img\\s+([^>]*)\\s*>";
+        return html.replaceAll(reg, "");
+    }
+
+    /*
+    public static void main(String[] args) {
+        String json = "{\"id\":1,\"answer\":null,\"question\":{\"id\":2,\"answer\":null,\"question\":{\"id\":3,\"answer\":null,\"question\":{\"id\":4,\"answer\":null,,\"question\":{\"id\":5,\"answer\":null}}}}}";
+        String result = HttpUtils.filterNullAttributes(json);
+        System.out.println(result);
+        String html = "111< img src='' >222< img / >333";
+        System.out.println(replaceImages(html));
+    }
+    */
+
+}

+ 224 - 0
src/main/java/cn/com/qmth/examcloud/app/core/utils/JsonMapper.java

@@ -0,0 +1,224 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 15:00:21.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.utils;
+
+import com.fasterxml.jackson.annotation.JsonInclude.Include;
+import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.core.type.TypeReference;
+import com.fasterxml.jackson.databind.*;
+import com.fasterxml.jackson.databind.util.JSONPObject;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.io.IOException;
+import java.util.Collection;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+/**
+ * 简单封装Jackson,实现JSON 与 Java Object互相转换的Mapper
+ * 封装不同的输出风格, 使用不同的builder函数创建实例
+ */
+@SuppressWarnings("unchecked")
+public class JsonMapper {
+    private static Logger log = LoggerFactory.getLogger(JsonMapper.class);
+    private ObjectMapper mapper;
+
+    public JsonMapper() {
+        this(null);
+    }
+
+    public JsonMapper(Include include) {
+        mapper = new ObjectMapper();
+        //设置输出时包含属性的风格
+        if (include != null) {
+            mapper.setSerializationInclusion(include);
+        }
+        //设置输入时忽略在JSON字符串中存在但Java对象实际没有的属性
+        mapper.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
+    }
+
+    /**
+     * 创建只输出非Null且非Empty(如List.isEmpty)的属性到Json字符串的Mapper,建议在外部接口中使用
+     */
+    public static JsonMapper nonEmptyMapper() {
+        return new JsonMapper(Include.NON_EMPTY);
+    }
+
+    public static JsonMapper nonNullMapper() {
+        return new JsonMapper(Include.NON_NULL);
+    }
+
+    /**
+     * 创建只输出初始值被改变的属性到Json字符串的Mapper, 最节约的存储方式,建议在内部接口中使用
+     */
+    public static JsonMapper nonDefaultMapper() {
+        return new JsonMapper(Include.NON_DEFAULT);
+    }
+
+    /**
+     * Object可以是POJO,也可以是Collection或数组
+     * 如果对象为Null, 返回"null"
+     * 如果集合为空集合, 返回"[]"
+     */
+    public String toJson(Object object) {
+        try {
+            return mapper.writeValueAsString(object);
+        } catch (IOException e) {
+            log.error("write to json string error:" + object);
+            return null;
+        }
+    }
+
+    /**
+     * 反序列化POJO或简单Collection如List<String>
+     * 如果JSON字符串为Null或"null"字符串, 返回Null
+     * 如果JSON字符串为"[]", 返回空集合
+     * 如需反序列化复杂Collection如List<MyBean>, 请使用fromJson(String, JavaType)
+     */
+    public <T> T fromJson(String jsonString, Class<T> clazz) {
+        if (StringUtils.isEmpty(jsonString)) {
+            return null;
+        }
+        try {
+            return mapper.readValue(jsonString, clazz);
+        } catch (IOException e) {
+            log.error("parse json string error", e);
+            return null;
+        }
+    }
+
+    /**
+     * 反序列化复杂Collection如List<Bean>, 先使用createCollectionType()或constructMapType()构造类型, 然后调用本函数
+     */
+    public <T> T fromJson(String jsonString, JavaType javaType) {
+        if (StringUtils.isEmpty(jsonString)) {
+            return null;
+        }
+        try {
+            return (T) mapper.readValue(jsonString, javaType);
+        } catch (IOException e) {
+            log.error("parse json string error", e);
+            return null;
+        }
+    }
+
+    /**
+     * 反序列化复杂的对象,如Page<Bean>
+     */
+    public <T> T fromJson(String jsonString, TypeReference javaType) {
+        if (StringUtils.isEmpty(jsonString)) {
+            return null;
+        }
+        try {
+            return (T) mapper.readValue(jsonString, javaType);
+        } catch (IOException e) {
+            log.error("parse json string error", e);
+            return null;
+        }
+    }
+
+    /**
+     * json to list
+     */
+    public <T> List<T> toList(String jsonString, Class<T> bean) {
+        if (StringUtils.isEmpty(jsonString)) {
+            return null;
+        }
+        try {
+            JavaType javaType = constructCollectionType(List.class, bean);
+            return mapper.readValue(jsonString, javaType);
+        } catch (IOException e) {
+            log.error("parse json string error", e);
+            return null;
+        }
+    }
+
+    /**
+     * json to simple HashMap
+     */
+    public <T> Map<String, T> toHashMap(String jsonString, Class<T> bean) {
+        if (StringUtils.isEmpty(jsonString)) {
+            return null;
+        }
+        try {
+            JavaType javaType = constructMapType(HashMap.class, String.class, bean);
+            return mapper.readValue(jsonString, javaType);
+        } catch (IOException e) {
+            log.error("parse json string error:", e);
+            return null;
+        }
+    }
+
+    /**
+     * 构造Collection类型
+     */
+    public JavaType constructCollectionType(Class<? extends Collection> collectionClass, Class<?> elementClass) {
+        return mapper.getTypeFactory().constructCollectionType(collectionClass, elementClass);
+    }
+
+    /**
+     * 构造Map类型
+     */
+    public JavaType constructMapType(Class<? extends Map> mapClass, Class<?> keyClass, Class<?> valueClass) {
+        return mapper.getTypeFactory().constructMapType(mapClass, keyClass, valueClass);
+    }
+
+    /**
+     * 当JSON里只含有Bean的部分屬性時,更新一個已存在Bean,只覆盖該部分的屬性
+     */
+    public void update(String jsonString, Object object) {
+        try {
+            mapper.readerForUpdating(object).readValue(jsonString);
+        } catch (JsonProcessingException e) {
+            log.error("update json string:" + jsonString + " to object:" + object + " error.");
+        } catch (IOException e) {
+            log.error("update json string:" + jsonString + " to object:" + object + " error.");
+        }
+    }
+
+    /**
+     * 輸出JSONP格式数据
+     */
+    public String toJsonP(String functionName, Object object) {
+        return toJson(new JSONPObject(functionName, object));
+    }
+
+    /**
+     * 設定是否使用Enum的toString函數來读写Enum
+     * 為False時使用Enum的name()函數來读写Enum, 默認為False
+     * 注意本函數一定要在Mapper創建後, 所有的读写動作之前調用
+     */
+    public void enableEnumUseToString() {
+        mapper.enable(SerializationFeature.WRITE_ENUMS_USING_TO_STRING);
+        mapper.enable(DeserializationFeature.READ_ENUMS_USING_TO_STRING);
+    }
+
+    /**
+     * 取出Mapper做进一步的设置或使用其他序列化API
+     */
+    public ObjectMapper getMapper() {
+        return mapper;
+    }
+
+    /***
+     * 把Json字符串转换成Node对象
+     */
+    public JsonNode getNode(String jsonStr) {
+        try {
+            //mapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_CONTROL_CHARS, true);
+            return mapper.readTree(jsonStr);
+        } catch (IOException e) {
+            log.error(e.getMessage(), e);
+            return null;
+        }
+    }
+
+}

+ 55 - 0
src/main/java/cn/com/qmth/examcloud/app/core/utils/StrUtils.java

@@ -0,0 +1,55 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-23 15:05:46.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.core.utils;
+
+import org.apache.commons.codec.digest.DigestUtils;
+
+import java.util.Random;
+import java.util.UUID;
+
+public class StrUtils {
+    private static Random random = new Random();
+
+    public static String uuid() {
+        return UUID.randomUUID().toString().replaceAll("-", "");
+    }
+
+    public static Integer randomNumber() {
+        return randomNumber(100000, 999999);
+    }
+
+    public static int randomNumber(int min, int max) {
+        return random.nextInt(max) % (max - min + 1) + min;
+    }
+
+    /**
+     * 校验手机号码格式
+     */
+    public static boolean isMobile(String str) {
+        return check(str, "^[1][3,4,5,6,7,8,9][0-9]{9}$");
+    }
+
+    public static boolean check(String str, String reg) {
+        if (str != null && str.matches(reg)) {
+            return true;
+        }
+        return false;
+    }
+
+    /**
+     * 将字符串MD5
+     */
+    public static String md5Key(String str) {
+        if (str == null) {
+            return null;
+        }
+        //MD5后字符串长度默认32位
+        return DigestUtils.md5Hex(str);
+    }
+
+}

+ 94 - 0
src/main/java/cn/com/qmth/examcloud/app/core/utils/ThreadUtils.java

@@ -0,0 +1,94 @@
+package cn.com.qmth.examcloud.app.core.utils;
+
+import com.google.common.collect.Maps;
+import org.apache.commons.lang3.StringUtils;
+
+import java.util.Map;
+import java.util.UUID;
+
+/**
+ * 线程工具
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public class ThreadUtils {
+
+    /**
+     * 属性注释
+     */
+    private static final ThreadLocal<String> LOCAL_TRACE_ID = new ThreadLocal<String>() {
+        @Override
+        public String initialValue() {
+            return null;
+        }
+    };
+
+    /**
+     * 属性注释
+     */
+    private static final ThreadLocal<Map<String, Object>> LOCAL_TRACE_MAP = new ThreadLocal<Map<String, Object>>() {
+        @Override
+        public Map<String, Object> initialValue() {
+            return null;
+        }
+    };
+
+    /**
+     * 重新初始化
+     */
+    public static String next() {
+        LOCAL_TRACE_MAP.set(null);
+        String traceID = UUID.randomUUID().toString().replace("-", "");
+        setTraceID(traceID);
+        return traceID;
+    }
+
+    /**
+     * 方法注释
+     */
+    public static String getTraceID() {
+        String traceID = LOCAL_TRACE_ID.get();
+        if (traceID == null) {
+            traceID = next();
+        } else {
+            return traceID;
+        }
+
+        return traceID;
+    }
+
+    /**
+     * 方法注释
+     */
+    public static void setTraceID(String traceID) {
+        if (!(StringUtils.isNotBlank(traceID)))
+            return;
+        LOCAL_TRACE_ID.set(traceID);
+    }
+
+    /**
+     * 方法注释
+     */
+    public static void set(String key, Object value) {
+        Map<String, Object> map = LOCAL_TRACE_MAP.get();
+
+        if (null == map) {
+            map = Maps.newHashMap();
+            LOCAL_TRACE_MAP.set(map);
+        }
+        map.put(key, value);
+    }
+
+    /**
+     * 方法注释
+     */
+    public static Object get(String key) {
+        Map<String, Object> map = LOCAL_TRACE_MAP.get();
+        if (null == map) {
+            return null;
+        }
+        return map.get(key);
+    }
+
+}

+ 20 - 0
src/main/java/cn/com/qmth/examcloud/app/dao/DeviceRecordRepository.java

@@ -0,0 +1,20 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 11:35:31.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.dao;
+
+import cn.com.qmth.examcloud.app.model.DeviceRecord;
+import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
+import org.springframework.data.repository.PagingAndSortingRepository;
+
+/**
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public interface DeviceRecordRepository extends PagingAndSortingRepository<DeviceRecord, Long>, JpaSpecificationExecutor<DeviceRecord> {
+
+}

+ 73 - 0
src/main/java/cn/com/qmth/examcloud/app/model/Constants.java

@@ -0,0 +1,73 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 15:00:21.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+/**
+ * 系统常量
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public interface Constants {
+
+    /**
+     * APP端的Token默认过期时间(秒)
+     * 默认一个月
+     */
+    int APP_SESSION_EXPIRE_TIME = 3600 * 24 * 30;
+
+    /**
+     * APP端用户登录信息的redis key前缀
+     */
+    String APP_SESSION_USER_KEY_PREFIX = "APP:SESSION_";
+
+    /**
+     * 平台端的Token默认过期时间(秒)
+     * 默认一小时
+     */
+    int PLATFORM_SESSION_EXPIRE_TIME = 3600;
+
+    /**
+     * 平台端的Token默认过期时间的redis key
+     */
+    String PLATFORM_SESSION_TIMEOUT_KEY = "$_SESSION_TIMEOUT";
+
+    /**
+     * 请求格式与默认编码集
+     */
+    String CHARSET_JSON_UTF8 = "application/json; charset=utf-8";
+
+    /* 默认请求的Header参数 */
+    String PARAM_KEY = "key";
+    String PARAM_TOKEN = "token";
+    String PARAM_APP_KEY = "app_key";
+    String PARAM_APP_TOKEN = "app_token";
+    String PARAM_DEVICE_ID = "deviceId";
+    String PARAM_TRACE_ID = "TRACE_ID";
+    String PARAM_CLIENT = "$spring_cloud_client";
+    String PARAM_CLIENT_VALUE = "-";
+
+    String PARAM_ROOT_ORG_ID = "rootOrgId";
+    String PARAM_ACCOUNT_TYPE = "accountType";
+    String PARAM_ACCOUNT = "account";
+    String PARAM_PSW = "password";
+
+    /* 常用的文件类型 */
+    String FILE_TYPE_ZIP = "zip";
+    String FILE_TYPE_PDF = "pdf";
+
+    /* 系统状态码 */
+    String CODE_0 = "0";//错误
+    String CODE_200 = "200";//成功
+    String CODE_500 = "500";//失败
+    String CODE_403 = "403";//认证失败
+    String CODE_404 = "404";//请求地址不存在
+    String CODE_P001012 = "P-001012";//认证失败
+    String CODE_B001012 = "B-001012";//认证失败
+
+}

+ 178 - 0
src/main/java/cn/com/qmth/examcloud/app/model/DeviceRecord.java

@@ -0,0 +1,178 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 09:57:01.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import com.fasterxml.jackson.annotation.JsonFormat;
+
+import javax.persistence.Column;
+import javax.persistence.Entity;
+import javax.persistence.Table;
+import java.util.Date;
+
+/**
+ * 设备访问记录
+ * 注:用于记录APP端调用接口时header中附带来源信息
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Entity
+@Table(name = "app_device_record")
+public class DeviceRecord extends IdEntity {
+    @Column(length = 50)
+    private String system;//系统标识,如:ios,android
+    @Column(length = 50)
+    private String deviceId;//设备编号
+    @Column(length = 50)
+    private String netType;//网络类型,如:WIFI,GPRS,2G,3G,4G,5G
+    @Column(length = 50)
+    private String brand;//设备品牌,如:iPhone,小米,华为
+    @Column(length = 50)
+    private String model;//型号,如:iPhone6 plus
+    @Column(length = 50)
+    private String sysVersion;//系统版本号,如:android6.0,ios11.0.2
+    @Column(length = 50)
+    private String appVersion;//app版本号,如:1.0.1,2.0.0
+    @Column(length = 50)
+    private String appCode;//app更新号[整数]
+    @Column(length = 50)
+    private String patchCode;//app补丁号
+    @Column(length = 50)
+    private String account;//来源账号(登录名或手机号)
+    @Column(length = 50)
+    private String loginKey;//登录认证KEY
+    @Column(length = 50)
+    private String loginToken;//登录认证TOKEN
+    @Column(length = 300)
+    private String url;//访问地址
+    @Column(length = 50)
+    private String ip;//访问IP
+    private Date createDate;//创建时间
+
+    public String getSystem() {
+        return system;
+    }
+
+    public void setSystem(String system) {
+        this.system = system;
+    }
+
+    public String getDeviceId() {
+        return deviceId;
+    }
+
+    public void setDeviceId(String deviceId) {
+        this.deviceId = deviceId;
+    }
+
+    public String getNetType() {
+        return netType;
+    }
+
+    public void setNetType(String netType) {
+        this.netType = netType;
+    }
+
+    public String getBrand() {
+        return brand;
+    }
+
+    public void setBrand(String brand) {
+        this.brand = brand;
+    }
+
+    public String getModel() {
+        return model;
+    }
+
+    public void setModel(String model) {
+        this.model = model;
+    }
+
+    public String getSysVersion() {
+        return sysVersion;
+    }
+
+    public void setSysVersion(String sysVersion) {
+        this.sysVersion = sysVersion;
+    }
+
+    public String getAppVersion() {
+        return appVersion;
+    }
+
+    public void setAppVersion(String appVersion) {
+        this.appVersion = appVersion;
+    }
+
+    public String getAppCode() {
+        return appCode;
+    }
+
+    public void setAppCode(String appCode) {
+        this.appCode = appCode;
+    }
+
+    public String getPatchCode() {
+        return patchCode;
+    }
+
+    public void setPatchCode(String patchCode) {
+        this.patchCode = patchCode;
+    }
+
+    public String getAccount() {
+        return account;
+    }
+
+    public void setAccount(String account) {
+        this.account = account;
+    }
+
+    public String getLoginKey() {
+        return loginKey;
+    }
+
+    public void setLoginKey(String loginKey) {
+        this.loginKey = loginKey;
+    }
+
+    public String getLoginToken() {
+        return loginToken;
+    }
+
+    public void setLoginToken(String loginToken) {
+        this.loginToken = loginToken;
+    }
+
+    public String getUrl() {
+        return url;
+    }
+
+    public void setUrl(String url) {
+        this.url = url;
+    }
+
+    public String getIp() {
+        return ip;
+    }
+
+    public void setIp(String ip) {
+        this.ip = ip;
+    }
+
+    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+08:00")
+    public Date getCreateDate() {
+        return createDate;
+    }
+
+    public void setCreateDate(Date createDate) {
+        this.createDate = createDate;
+    }
+
+}

+ 42 - 0
src/main/java/cn/com/qmth/examcloud/app/model/DeviceRecordQuery.java

@@ -0,0 +1,42 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-01 11:45:59.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+/**
+ * 设备访问记录查询类
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public class DeviceRecordQuery extends DeviceRecord {
+    private Integer pageSize;
+    private Integer pageNo;
+
+    public Integer getPageSize() {
+        if (pageSize == null || pageSize < 1) {
+            pageSize = 1;
+        }
+        return pageSize;
+    }
+
+    public void setPageSize(Integer pageSize) {
+        this.pageSize = pageSize;
+    }
+
+    public Integer getPageNo() {
+        if (pageNo == null || pageNo < 1) {
+            pageNo = 1;
+        }
+        return pageNo;
+    }
+
+    public void setPageNo(Integer pageNo) {
+        this.pageNo = pageNo;
+    }
+
+}

+ 27 - 0
src/main/java/cn/com/qmth/examcloud/app/model/ExamInfo.java

@@ -0,0 +1,27 @@
+/*
+ * *************************************************
+ * Copyright (c) 2019 QMTH. All Rights Reserved.
+ * Created by Deason on 2019-05-17 15:35:06.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import java.io.Serializable;
+
+public class ExamInfo implements Serializable {
+    private static final long serialVersionUID = 5830281415991391915L;
+    /**
+     * 是否存在正在进行中的考试
+     */
+    private Boolean existExamingRecord;
+
+    public Boolean getExistExamingRecord() {
+        return existExamingRecord != null ? existExamingRecord : false;
+    }
+
+    public void setExistExamingRecord(Boolean existExamingRecord) {
+        this.existExamingRecord = existExamingRecord;
+    }
+
+}

+ 68 - 0
src/main/java/cn/com/qmth/examcloud/app/model/GetYunSignatureReq.java

@@ -0,0 +1,68 @@
+package cn.com.qmth.examcloud.app.model;
+
+import cn.com.qmth.examcloud.api.commons.exchange.JsonSerializable;
+import io.swagger.annotations.ApiModelProperty;
+
+import javax.validation.constraints.NotNull;
+
+public class GetYunSignatureReq implements JsonSerializable {
+	/**
+     * 
+     */
+    private static final long serialVersionUID = 4523536593807014828L;
+    @NotNull(message = "考试记录DataID不能为空")
+    @ApiModelProperty(required = true, value = "考试记录DataID")
+    private String examRecordDataId;
+    @NotNull(message = "题号不能为空")
+    @ApiModelProperty(required = true, value = "考试试题号")
+    private String order;
+    @NotNull(message = "文件MD5不能为空")
+    @ApiModelProperty(required = true, value = "文件MD5")
+    private String fileMd5;
+    @NotNull(message = "文件后缀不能为空")
+    @ApiModelProperty(required = true, value = "文件后缀")
+    private String fileSuffix;
+    @ApiModelProperty(value = "文件名自定义参数")
+    private String ext;
+
+    public String getExamRecordDataId() {
+        return examRecordDataId;
+    }
+
+    public void setExamRecordDataId(String examRecordDataId) {
+        this.examRecordDataId = examRecordDataId;
+    }
+
+    public String getOrder() {
+        return order;
+    }
+
+    public void setOrder(String order) {
+        this.order = order;
+    }
+
+    public String getFileMd5() {
+        return fileMd5;
+    }
+
+    public void setFileMd5(String fileMd5) {
+        this.fileMd5 = fileMd5;
+    }
+
+    public String getFileSuffix() {
+        return fileSuffix;
+    }
+
+    public void setFileSuffix(String fileSuffix) {
+        this.fileSuffix = fileSuffix;
+    }
+
+    public String getExt() {
+        return ext;
+    }
+
+    public void setExt(String ext) {
+        this.ext = ext;
+    }
+
+}

+ 31 - 0
src/main/java/cn/com/qmth/examcloud/app/model/IdEntity.java

@@ -0,0 +1,31 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-20 11:24:17.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import javax.persistence.GeneratedValue;
+import javax.persistence.GenerationType;
+import javax.persistence.Id;
+import javax.persistence.MappedSuperclass;
+import java.io.Serializable;
+
+@MappedSuperclass
+public abstract class IdEntity implements Serializable {
+    protected static final long serialVersionUID = 1L;
+    @Id
+    @GeneratedValue(strategy = GenerationType.IDENTITY)
+    protected Long id;//主键ID
+
+    public Long getId() {
+        return id;
+    }
+
+    public void setId(Long id) {
+        this.id = id;
+    }
+
+}

+ 194 - 0
src/main/java/cn/com/qmth/examcloud/app/model/LoginInfo.java

@@ -0,0 +1,194 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 14:32:47.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import java.io.Serializable;
+import java.util.Calendar;
+import java.util.Date;
+
+/**
+ * 用户登录信息
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public class LoginInfo implements Serializable {
+    private static final long serialVersionUID = 1L;
+    private Long userId;
+    private String userName;
+    private String account;
+    private String password;
+    private String accountType;
+    private Long rootOrgId;
+    private String domain;
+    private String key;
+    private String token;
+    private String appToken;
+    private String deviceId;
+    private Date createTime;
+    private String smsCode;
+    private Boolean noSession;
+
+    public LoginInfo(String account, String password, String accountType, Long rootOrgId, String domain, String key, String token, String deviceId) {
+        this.account = account;
+        this.password = password;
+        this.accountType = accountType;
+        this.rootOrgId = rootOrgId;
+        this.domain = domain;
+        this.key = key;
+        this.token = token;
+        this.deviceId = deviceId;
+        this.createTime = new Date();
+    }
+
+    public LoginInfo(String account, String password, String accountType, Long rootOrgId, String domain, String deviceId, String smsCode) {
+        this.account = account;
+        this.password = password;
+        this.accountType = accountType;
+        this.rootOrgId = rootOrgId;
+        this.domain = domain;
+        this.deviceId = deviceId;
+        this.createTime = new Date();
+        this.smsCode = smsCode;
+    }
+
+    public LoginInfo() {
+        this.createTime = new Date();
+    }
+
+    public boolean hasExpired(int seconds) {
+        if (createTime == null) {
+            return true;
+        }
+
+        Calendar c = Calendar.getInstance();
+        c.setTime(createTime);
+        c.add(Calendar.SECOND, seconds - 60);
+        //System.out.println(DateUtils.format(c.getTime()));
+
+        //判断是否在n小时内
+        if (c.getTime().after(new Date())) {
+            return false;
+        }
+
+        return true;
+    }
+
+    public Long getUserId() {
+        return userId;
+    }
+
+    public void setUserId(Long userId) {
+        this.userId = userId;
+    }
+
+    public String getUserName() {
+        return userName;
+    }
+
+    public void setUserName(String userName) {
+        this.userName = userName;
+    }
+
+    public String getAccount() {
+        return account;
+    }
+
+    public void setAccount(String account) {
+        this.account = account;
+    }
+
+    public String getPassword() {
+        return password;
+    }
+
+    public void setPassword(String password) {
+        this.password = password;
+    }
+
+    public String getAccountType() {
+        return accountType;
+    }
+
+    public void setAccountType(String accountType) {
+        this.accountType = accountType;
+    }
+
+    public Long getRootOrgId() {
+        return rootOrgId;
+    }
+
+    public void setRootOrgId(Long rootOrgId) {
+        this.rootOrgId = rootOrgId;
+    }
+
+    public String getDomain() {
+        return domain;
+    }
+
+    public void setDomain(String domain) {
+        this.domain = domain;
+    }
+
+    public String getKey() {
+        return key;
+    }
+
+    public void setKey(String key) {
+        this.key = key;
+    }
+
+    public String getToken() {
+        return token;
+    }
+
+    public void setToken(String token) {
+        this.token = token;
+    }
+
+    public String getAppToken() {
+        return appToken;
+    }
+
+    public void setAppToken(String appToken) {
+        this.appToken = appToken;
+    }
+
+    public String getDeviceId() {
+        return deviceId;
+    }
+
+    public void setDeviceId(String deviceId) {
+        this.deviceId = deviceId;
+    }
+
+    public Date getCreateTime() {
+        return createTime;
+    }
+
+    public void setCreateTime(Date createTime) {
+        this.createTime = createTime;
+    }
+
+    public String getSmsCode() {
+        return smsCode;
+    }
+
+    public void setSmsCode(String smsCode) {
+        this.smsCode = smsCode;
+    }
+
+    public Boolean getNoSession() {
+        return noSession;
+    }
+
+    public void setNoSession(Boolean noSession) {
+        this.noSession = noSession;
+    }
+
+}

+ 38 - 0
src/main/java/cn/com/qmth/examcloud/app/model/LoginType.java

@@ -0,0 +1,38 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 16:20:47.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+/**
+ * 登录类型枚举类
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public enum LoginType {
+
+    /**
+     * 常规登录名(非学生登录)
+     */
+    COMMON_LOGIN_NAME,
+
+    /**
+     * 学生学号登录
+     */
+    STUDENT_CODE,
+
+    /**
+     * 学生手机号登录
+     */
+    STUDENT_PHONE,
+
+    /**
+     * 学生身份证号登录
+     */
+    STUDENT_IDENTITY_NUMBER
+
+}

+ 74 - 0
src/main/java/cn/com/qmth/examcloud/app/model/ResBody.java

@@ -0,0 +1,74 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-17 16:14:47.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import java.io.Serializable;
+
+public class ResBody implements Serializable {
+    private static final long serialVersionUID = 1L;
+    /* 考务、题库等接口返回结果 */
+    private String code;//状态码
+    private String desc;//描述信息
+
+    /* 网考接口返回结果 */
+    private String errorMsg;
+    private String content;
+
+    /* 短信接口返回结果 */
+    private Boolean success;
+    private String returnMsg;
+
+    public String getCode() {
+        return code;
+    }
+
+    public void setCode(String code) {
+        this.code = code;
+    }
+
+    public String getDesc() {
+        return desc;
+    }
+
+    public void setDesc(String desc) {
+        this.desc = desc;
+    }
+
+    public String getErrorMsg() {
+        return errorMsg;
+    }
+
+    public void setErrorMsg(String errorMsg) {
+        this.errorMsg = errorMsg;
+    }
+
+    public String getContent() {
+        return content;
+    }
+
+    public void setContent(String content) {
+        this.content = content;
+    }
+
+    public Boolean getSuccess() {
+        return success;
+    }
+
+    public void setSuccess(Boolean success) {
+        this.success = success;
+    }
+
+    public String getReturnMsg() {
+        return returnMsg;
+    }
+
+    public void setReturnMsg(String returnMsg) {
+        this.returnMsg = returnMsg;
+    }
+
+}

+ 119 - 0
src/main/java/cn/com/qmth/examcloud/app/model/Result.java

@@ -0,0 +1,119 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-16 15:00:21.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import com.fasterxml.jackson.annotation.JsonIgnore;
+import org.apache.commons.lang3.builder.ToStringBuilder;
+import org.apache.commons.lang3.builder.ToStringStyle;
+import springfox.documentation.annotations.ApiIgnore;
+
+import java.io.Serializable;
+
+public class Result<T> implements Serializable {
+    private static final long serialVersionUID = 1L;
+    private String code;//状态码
+    private String desc;//描述信息
+    private T data;//数据
+
+    public static final String DESC_200 = "请求成功!";
+    public static final String DESC_500 = "请求失败!";
+    public static final String DESC_403 = "无效认证信息,请先登录!";
+    public static final String DESC_404 = "请求地址或参数错误!";
+
+    public Result() {
+
+    }
+
+    public Result(String code, String desc, T data) {
+        this.code = code;
+        this.desc = desc;
+        this.data = data;
+    }
+
+    /**
+     * 成功结果
+     */
+    public Result success(T data) {
+        this.code = Constants.CODE_200;
+        this.desc = DESC_200;
+        this.data = data;
+        return this;
+    }
+
+    public Result success() {
+        this.code = Constants.CODE_200;
+        this.desc = DESC_200;
+        return this;
+    }
+
+    /**
+     * 错误结果
+     */
+    public Result error(String desc) {
+        this.code = Constants.CODE_500;
+        this.desc = desc;
+        return this;
+    }
+
+    public Result error() {
+        this.code = Constants.CODE_500;
+        this.desc = DESC_500;
+        return this;
+    }
+
+    public Result noAuthError() {
+        this.code = Constants.CODE_403;
+        this.desc = DESC_403;
+        return this;
+    }
+
+    public Result noFoundError() {
+        this.code = Constants.CODE_404;
+        this.desc = DESC_404;
+        return this;
+    }
+
+    @JsonIgnore
+    @ApiIgnore
+    public boolean isSuccess() {
+        if (Constants.CODE_200.equals(getCode())) {
+            return true;
+        }
+        return false;
+    }
+
+    public String getCode() {
+        return code;
+    }
+
+    public String getDesc() {
+        return desc;
+    }
+
+    public T getData() {
+        return data;
+    }
+
+    public void setCode(String code) {
+        this.code = code;
+    }
+
+    public void setDesc(String desc) {
+        this.desc = desc;
+    }
+
+    public void setData(T data) {
+        this.data = data;
+    }
+
+    @Override
+    public String toString() {
+        return ToStringBuilder.reflectionToString(this, ToStringStyle.JSON_STYLE);
+    }
+
+}

+ 223 - 0
src/main/java/cn/com/qmth/examcloud/app/model/UserInfo.java

@@ -0,0 +1,223 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 14:32:37.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.model;
+
+import java.io.Serializable;
+import java.util.List;
+
+/**
+ * 用户信息
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+public class UserInfo implements Serializable {
+    private static final long serialVersionUID = 8766713125414955078L;
+    private Long id;
+    /**
+     * 用户ID
+     */
+    private Long userId;
+    /**
+     * 用户类型
+     */
+    private String userType;
+    /**
+     * 用户名
+     */
+    private String displayName;
+    /**
+     * 身份证号
+     */
+    private String identityNumber;
+    /**
+     * 学号
+     */
+    private String studentCode;
+    /**
+     * 学号列表
+     */
+    private List<String> studentCodeList;
+    /**
+     * 学生ID
+     */
+    private String studentId;
+    /**
+     * 登录手机号
+     */
+    private String securityPhone;
+    /**
+     * 手机号
+     */
+    private String phoneNumber;
+    /**
+     * 图像地址
+     */
+    private String photoPath;
+    /**
+     * 顶级机构ID
+     */
+    private Long rootOrgId;
+    /**
+     * 顶级机构(学校)名称
+     */
+    private String rootOrgName;
+    /**
+     * 学习中心ID
+     */
+    private Long orgId;
+    /**
+     * 学习中心名称
+     */
+    private String orgName;
+    /**
+     * 全局唯一标识符
+     */
+    private String key;
+    /**
+     * 鉴权token
+     */
+    private String token;
+
+    public Long getId() {
+        return id;
+    }
+
+    public void setId(Long id) {
+        this.id = id;
+    }
+
+    public Long getUserId() {
+        return userId;
+    }
+
+    public void setUserId(Long userId) {
+        this.userId = userId;
+    }
+
+    public String getUserType() {
+        return userType;
+    }
+
+    public void setUserType(String userType) {
+        this.userType = userType;
+    }
+
+    public String getDisplayName() {
+        return displayName;
+    }
+
+    public void setDisplayName(String displayName) {
+        this.displayName = displayName;
+    }
+
+    public String getIdentityNumber() {
+        return identityNumber;
+    }
+
+    public void setIdentityNumber(String identityNumber) {
+        this.identityNumber = identityNumber;
+    }
+
+    public String getStudentCode() {
+        return studentCode;
+    }
+
+    public void setStudentCode(String studentCode) {
+        this.studentCode = studentCode;
+    }
+
+    public String getStudentId() {
+        return studentId;
+    }
+
+    public void setStudentId(String studentId) {
+        this.studentId = studentId;
+    }
+
+    public List<String> getStudentCodeList() {
+        return studentCodeList;
+    }
+
+    public void setStudentCodeList(List<String> studentCodeList) {
+        this.studentCodeList = studentCodeList;
+    }
+
+    public String getSecurityPhone() {
+        return securityPhone;
+    }
+
+    public void setSecurityPhone(String securityPhone) {
+        this.securityPhone = securityPhone;
+    }
+
+    public String getPhoneNumber() {
+        return phoneNumber;
+    }
+
+    public void setPhoneNumber(String phoneNumber) {
+        this.phoneNumber = phoneNumber;
+    }
+
+    public String getPhotoPath() {
+        return photoPath;
+    }
+
+    public void setPhotoPath(String photoPath) {
+        this.photoPath = photoPath;
+    }
+
+    public Long getRootOrgId() {
+        return rootOrgId;
+    }
+
+    public void setRootOrgId(Long rootOrgId) {
+        this.rootOrgId = rootOrgId;
+    }
+
+    public String getRootOrgName() {
+        return rootOrgName;
+    }
+
+    public void setRootOrgName(String rootOrgName) {
+        this.rootOrgName = rootOrgName;
+    }
+
+    public Long getOrgId() {
+        return orgId;
+    }
+
+    public void setOrgId(Long orgId) {
+        this.orgId = orgId;
+    }
+
+    public String getOrgName() {
+        return orgName;
+    }
+
+    public void setOrgName(String orgName) {
+        this.orgName = orgName;
+    }
+
+    public String getKey() {
+        return key;
+    }
+
+    public void setKey(String key) {
+        this.key = key;
+    }
+
+    public String getToken() {
+        return token;
+    }
+
+    public void setToken(String token) {
+        this.token = token;
+    }
+
+}

+ 140 - 0
src/main/java/cn/com/qmth/examcloud/app/service/CoreAuthService.java

@@ -0,0 +1,140 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:40:42.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.model.LoginInfo;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.model.UserInfo;
+
+/**
+ * 认证中心业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface CoreAuthService {
+
+    /**
+     * 检查某考生是否存在正在(在线考试类型)考试中
+     */
+    boolean isDoingExam(Long rootOrgId, String accountType, String account);
+
+    /**
+     * 检查学校是否开放APP功能
+     */
+    boolean isOpenApp(Long rootOrgId);
+
+    /**
+     * 根据手机号获取学校ID
+     */
+    Long getRootOrgIdBySecurityPhone(String securityPhone);
+
+    /**
+     * 用户登录
+     *
+     * @param loginInfo
+     * @return
+     * @throws Exception
+     */
+    Result<UserInfo> login(LoginInfo loginInfo);
+
+    /**
+     * 用户退出登录
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result logout(String key, String token) throws Exception;
+
+    /**
+     * 修改密码
+     *
+     * @param key
+     * @param token
+     * @param studentId
+     * @param password
+     * @param newPassword
+     * @return
+     * @throws Exception
+     */
+    Result updateStudentPassword(String key, String token, Long studentId, String password, String newPassword) throws Exception;
+
+    /**
+     * 重置密码
+     *
+     * @param key
+     * @param token
+     * @param newPassword
+     * @return
+     * @throws Exception
+     */
+    Result resetStudentPassword(String key, String token, String newPassword) throws Exception;
+
+    /**
+     * 获取短信验证码
+     *
+     * @param key
+     * @param token
+     * @param phone
+     * @return
+     * @throws Exception
+     */
+    Result sendSmsCode(String key, String token, String phone) throws Exception;
+
+    /**
+     * 获取短信验证码
+     *
+     * @param phone
+     * @return
+     * @throws Exception
+     */
+    Result code4Student(String phone, Boolean bound) throws Exception;
+
+    /**
+     * 保存用户绑定的手机号
+     *
+     * @param key
+     * @param token
+     * @param phone
+     * @param code
+     * @return
+     * @throws Exception
+     */
+    Result userBindingPhone(String key, String token, String phone, String code) throws Exception;
+
+    /**
+     * 缓存用户登录信息
+     *
+     * @param loginInfo
+     * @param key
+     */
+    void cacheLoginInfo(LoginInfo loginInfo, String key);
+
+    /**
+     * 获取缓存中的用户登录信息
+     *
+     * @param key
+     * @return
+     */
+    LoginInfo getLoginInfo(String key);
+
+    /**
+     * 获取平台端的默认过期时间(秒)
+     *
+     * @return
+     */
+    int getSessionTimeout();
+
+    /**
+     * 初始化内部接口请求鉴权
+     */
+    void initRequestTrace();
+
+}

+ 30 - 0
src/main/java/cn/com/qmth/examcloud/app/service/CoreBasicService.java

@@ -0,0 +1,30 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-02 15:15:46.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.model.Result;
+
+/**
+ * 基础信息服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface CoreBasicService {
+
+    /**
+     * 获取学生用户信息
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result getStudentInfo(String key, String token) throws Exception;
+
+}

+ 75 - 0
src/main/java/cn/com/qmth/examcloud/app/service/CoreExamWorkService.java

@@ -0,0 +1,75 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:29:12.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.model.Result;
+
+/**
+ * 考务业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface CoreExamWorkService {
+
+    /**
+     * 获取某考生的“考试批次”列表
+     *
+     * @param key
+     * @param token
+     * @param studentId
+     * @return
+     * @throws Exception
+     */
+    Result getPracticeExamList(String key, String token, String studentId) throws Exception;
+
+    /**
+     * 获取当前练习的考试基本信息
+     *
+     * @param key
+     * @param token
+     * @param examId
+     * @return
+     * @throws Exception
+     */
+    Result getExamInfo(String key, String token, Long examId) throws Exception;
+
+    /**
+     * 查询考试说明
+     *
+     * @param key
+     * @param token
+     * @param examId
+     * @return
+     * @throws Exception
+     */
+    Result getBeforeExamRemark(String key, String token, Long examId, String type) throws Exception;
+
+    /**
+     * 获取冻结时间
+     *
+     * @param key
+     * @param token
+     * @param examId
+     * @return
+     * @throws Exception
+     */
+    Result getFreezeTime(String key, String token, Long examId) throws Exception;
+
+    /**
+     * 获取文件上传类型
+     *
+     * @param key
+     * @param token
+     * @param examId
+     * @return
+     * @throws Exception
+     */
+    Result getUpLoadType(String key, String token, Long examId) throws Exception;
+
+}

+ 309 - 0
src/main/java/cn/com/qmth/examcloud/app/service/CoreOeService.java

@@ -0,0 +1,309 @@
+/*
+ * ************************************************* Copyright (c) 2018 QMTH.
+ * All Rights Reserved. Created by Deason on 2018-07-31 17:31:34.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.model.GetYunSignatureReq;
+import cn.com.qmth.examcloud.app.model.Result;
+
+/**
+ * 网考业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface CoreOeService {
+
+    /**
+     * 获取服务器当前时间
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result getCurrentTime(String key, String token) throws Exception;
+
+    /**
+     * 获取当前用户参加的离线课程列表
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result getOfflineExamCourseList(String key, String token) throws Exception;
+
+    /**
+     * 离线考试的抽取考题
+     *
+     * @param key
+     * @param token
+     * @param examStudentId
+     * @return
+     * @throws Exception
+     */
+    Result startOfflineExamRecord(String key, String token, String examStudentId) throws Exception;
+
+    /**
+     * 上传作答文件
+     *
+     * @param key
+     * @param token
+     * @param examRecordId
+     * @param fileBytes
+     * @param fileName
+     * @param md5
+     * @return
+     * @throws Exception
+     */
+    Result uploadPaperAnswer(String key, String token, String examRecordId, byte[] fileBytes, String fileName,
+            String md5) throws Exception;
+
+    /**
+     * 获取某考试批次下的课程列表
+     *
+     * @param key
+     * @param token
+     * @param examId
+     * @return
+     * @throws Exception
+     */
+    Result getPracticeExamCourseList(String key, String token, String examId) throws Exception;
+
+    /**
+     * 获取当前练习的剩余作答时间
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordHeartbeat(String key, String token) throws Exception;
+
+    /**
+     * 考生“开始练习”
+     *
+     * @param key
+     * @param token
+     * @param examStudentId
+     * @return
+     * @throws Exception
+     */
+    Result startPracticeExamRecord(String key, String token, String examStudentId) throws Exception;
+
+    /**
+     * 获取当前练习的试卷大题结构信息
+     *
+     * @param key
+     * @param token
+     * @param examRecordId
+     * @return
+     * @throws Exception
+     */
+    Result getAdminExamPaperStructList(String key, String token, String examRecordId, String fromCache)
+            throws Exception;
+
+    /**
+     * 获取当前练习的试卷大题结构信息
+     *
+     * @param key
+     * @param token
+     * @param examRecordId
+     * @return
+     * @throws Exception
+     */
+    Result getStudentExamPaperStructList(String key, String token, String examRecordId) throws Exception;
+
+    /**
+     * 获取当前练习的试卷试题列表
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordPaperQuestionList(String key, String token) throws Exception;
+
+    /**
+     * 获取考生作答的某个试题的详细信息
+     *
+     * @param key
+     * @param token
+     * @param questionId
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordPaperQuestionDetail(String key, String token, String questionId, String examRecordId)
+            throws Exception;
+
+    /**
+     * 保存或更新考生作答的某个试题答案
+     *
+     * @param key
+     * @param token
+     * @param studentAnswer
+     * @return
+     * @throws Exception
+     */
+    Result updateExamRecordPaperQuestionAnswer(String key, String token, Integer order, String studentAnswer)
+            throws Exception;
+
+    /**
+     * 当前练习的交卷
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result submitPracticeExamRecord(String key, String token) throws Exception;
+
+    /**
+     * 检查考生当前是否有正在进行的练习记录
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result checkOnlineExamRecord(String key, String token) throws Exception;
+
+    /**
+     * 获取当前考生的当前课程的历史练习记录
+     *
+     * @param key
+     * @param token
+     * @param examStudentId
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordPracticeHistoryList(String key, String token, String examStudentId) throws Exception;
+
+    /**
+     * 获取成绩报告的答题情况统计
+     *
+     * @param key
+     * @param token
+     * @param examRecordId
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordTotalInfo(String key, String token, String examRecordId, String fromCache) throws Exception;
+
+    /**
+     * 获取作答的题列表
+     *
+     * @param key
+     * @param token
+     * @param examRecordId
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordQuestionDetailList(String key, String token, String examRecordId) throws Exception;
+
+    /**
+     * 获取当前试题的音频已播放次数
+     *
+     * @param key
+     * @param token
+     * @param questionId
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordQuestionAudioPlayTimes(String key, String token, String questionId) throws Exception;
+
+    /**
+     * 更新当前试题的音频已播放次数
+     *
+     * @param key
+     * @param token
+     * @param questionId
+     * @param mediaName
+     * @return
+     * @throws Exception
+     */
+    Result updateExamRecordQuestionAudioPlayTimes(String key, String token, String questionId, String mediaName)
+            throws Exception;
+
+    /**
+     * 练习记录配置信息
+     *
+     * @param key
+     * @param token
+     * @param examRecordDataId
+     * @return
+     * @throws Exception
+     */
+    Result findExamRecordDataEntity(String key, String token, Long examRecordDataId, String fromCache) throws Exception;
+
+    /**
+     * 练习记录试题列表
+     *
+     * @param key
+     * @param token
+     * @param examRecordDataId
+     * @return
+     * @throws Exception
+     */
+    Result getExamRecordQuestions(String key, String token, Long examRecordDataId, String fromCache) throws Exception;
+
+    /**
+     * 查询考生的考试批次属性集
+     * 
+     * @param key
+     * @param token
+     * @param examId
+     * @param keys
+     * @return
+     * @throws Exception
+     */
+    Result getExamPropertyFromCacheByStudentSession(String key, String token, Long examId, String keys)
+            throws Exception;
+
+    /**
+     * 根据Id获取试卷
+     * 
+     * @param key
+     * @param token
+     * @param paperId
+     * @return
+     */
+    Result getPaperById(String key, String token, String paperId) throws Exception;
+
+    /**获取文件上传签名
+     * @param key
+     * @param token
+     * @param req
+     * @return
+     * @throws Exception
+     */
+    Result getYunSignature(String key, String token, GetYunSignatureReq req) throws Exception;
+
+    /**获取在线考试待考列表
+     * @param key
+     * @param token
+     * @return
+     */
+    Result queryExamList(String key, String token) throws Exception;
+
+    /**获取考试记录信息
+     * @param key
+     * @param token
+     * @param examRecordDataId
+     * @return
+     */
+    Result getEndExamInfo(String key, String token, Long examRecordDataId) throws Exception;
+
+    /**根据examStudentId获取客观分信息
+     * @param key
+     * @param token
+     * @param examStudentId
+     * @return
+     */
+    Result queryObjectiveScoreList(String key, String token, Long examStudentId) throws Exception;
+
+}

+ 51 - 0
src/main/java/cn/com/qmth/examcloud/app/service/CoreQuestionService.java

@@ -0,0 +1,51 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:34:46.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.model.Result;
+
+/**
+ * 题库业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface CoreQuestionService {
+
+    /**
+     * 下载考题
+     *
+     * @param paperId
+     * @param orgName
+     * @return
+     * @throws Exception
+     */
+    String downloadPaper(String key, String token, String paperId, String orgName) throws Exception;
+
+    /**
+     * 获取某份试卷的详细信息
+     *
+     * @param key
+     * @param token
+     * @param paperId
+     * @return
+     * @throws Exception
+     */
+    Result getPaperDetail(String key, String token, String paperId) throws Exception;
+
+    /**
+     * 获取某个试题的详细信息
+     *
+     * @param key
+     * @param token
+     * @return
+     * @throws Exception
+     */
+    Result getQuestion(String key, String token, String courseCode, Long examId, String groupCode, String questionId) throws Exception;
+
+}

+ 40 - 0
src/main/java/cn/com/qmth/examcloud/app/service/DeviceRecordService.java

@@ -0,0 +1,40 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:15:16.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.model.DeviceRecord;
+import cn.com.qmth.examcloud.app.model.DeviceRecordQuery;
+import cn.com.qmth.examcloud.app.model.Result;
+import org.springframework.data.domain.Page;
+
+/**
+ * 设备访问记录Service
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface DeviceRecordService {
+
+    /**
+     * 查询设备访问记录(分页)
+     *
+     * @param params
+     * @return
+     * @throws Exception
+     */
+    Result<Page<DeviceRecord>> getDeviceRecordList(DeviceRecordQuery params) throws Exception;
+
+    /**
+     * 新增设备访问记录
+     *
+     * @param deviceRecord
+     * @throws Exception
+     */
+    void addDeviceRecord(DeviceRecord deviceRecord) throws Exception;
+
+}

+ 82 - 0
src/main/java/cn/com/qmth/examcloud/app/service/RedisService.java

@@ -0,0 +1,82 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:53:48.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.core.utils.JsonMapper;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.data.redis.core.RedisTemplate;
+import org.springframework.data.redis.core.StringRedisTemplate;
+import org.springframework.stereotype.Service;
+
+import java.util.concurrent.TimeUnit;
+
+/**
+ * Redis接口服务类
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class RedisService {
+    private static Logger log = LoggerFactory.getLogger(RedisService.class);
+    @Autowired
+    private StringRedisTemplate stringRedisTemplate;
+    @Autowired
+    private RedisTemplate<String, Object> redisTemplate;
+
+    public void setObj(String key, Object value) {
+        redisTemplate.opsForValue().set(key, value);
+    }
+
+    public void setObj(String key, Object value, long seconds) {
+        redisTemplate.opsForValue().set(key, value, seconds, TimeUnit.SECONDS);
+    }
+
+    public void set(String key, String value) {
+        stringRedisTemplate.opsForValue().set(key, value);
+    }
+
+    public void set(String key, String value, long seconds) {
+        stringRedisTemplate.opsForValue().set(key, value, seconds, TimeUnit.SECONDS);
+    }
+
+    public String get(String key) {
+        return stringRedisTemplate.opsForValue().get(key);
+    }
+
+    public <T> T getObj(String key, Class<T> clazz) {
+        return (T) redisTemplate.opsForValue().get(key);
+    }
+
+    public <T> T fromJson(String key, Class<T> clazz) {
+        String jsonStr = stringRedisTemplate.opsForValue().get(key);
+        if (jsonStr != null) {
+            return new JsonMapper().fromJson(jsonStr, clazz);
+        }
+        return null;
+    }
+
+    public boolean exist(String key) {
+        return stringRedisTemplate.hasKey(key);
+    }
+
+    public void delete(String key) {
+        stringRedisTemplate.delete(key);
+    }
+
+    public StringRedisTemplate getStringRedisTemplate() {
+        return stringRedisTemplate;
+    }
+
+    public RedisTemplate<String, Object> getRedisTemplate() {
+        return redisTemplate;
+    }
+
+}

+ 147 - 0
src/main/java/cn/com/qmth/examcloud/app/service/RouterService.java

@@ -0,0 +1,147 @@
+package cn.com.qmth.examcloud.app.service;
+
+import cn.com.qmth.examcloud.app.core.SysProperty;
+import cn.com.qmth.examcloud.app.core.router.Router;
+import cn.com.qmth.examcloud.app.core.utils.HttpClientBuilder;
+import cn.com.qmth.examcloud.app.model.Constants;
+import cn.com.qmth.examcloud.app.model.Result;
+import okhttp3.*;
+import org.apache.commons.collections.MapUtils;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Component;
+import org.springframework.util.Assert;
+
+import java.io.UnsupportedEncodingException;
+import java.net.ConnectException;
+import java.net.URLEncoder;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Random;
+import java.util.UUID;
+
+import static cn.com.qmth.examcloud.app.model.Constants.*;
+
+@Component
+public class RouterService {
+    private static Logger log = LoggerFactory.getLogger(RouterService.class);
+    private Random random = new Random();
+    @Autowired
+    private SysProperty sysProperty;
+
+    public Result execute(Router router) {
+        //校验参数
+        Assert.notNull(router, "参数“router”不能为空!");
+        Assert.notNull(router.getServer(), "参数“server”不能为空!");
+        Assert.hasLength(router.getUrl(), "参数“url”不能为空!");
+        Assert.notNull(router.getMethod(), "参数“method”不能为空!");
+
+        //处理请求Headers
+        Map<String, String> requestHeaders;
+        if (MapUtils.isEmpty(router.getHeaders())) {
+            requestHeaders = new HashMap<>();
+        } else {
+            requestHeaders = router.getHeaders();
+        }
+
+        String traceId = this.getRandomTraceId();
+        requestHeaders.put(PARAM_KEY, router.getKey());
+        requestHeaders.put(PARAM_TOKEN, router.getToken());
+        requestHeaders.put(PARAM_TRACE_ID, traceId);
+        requestHeaders.put(PARAM_CLIENT, PARAM_CLIENT_VALUE);
+
+        //处理请求Params
+        String requestParams = this.appendParams(router.getParams());
+
+        //处理请求Body
+        RequestBody requestBody = this.appendBody(router.getBody());
+
+        //处理请求地址
+        final String requestUrl = sysProperty.getProxyUrl(router.getServer()) + "/" + router.getUrl() + requestParams;
+        log.info(String.format("[%s][%s][%s]%s ", traceId, router.getServer().getInstanceName(), router.getMethod().name(), requestUrl));
+
+        //封装请求
+        Request.Builder request = new Request.Builder()
+                .url(requestUrl)
+                .headers(Headers.of(requestHeaders));
+
+        switch (router.getMethod()) {
+            case GET:
+                return this.call(request.get().build(), traceId);
+            case POST:
+                return this.call(request.post(requestBody).build(), traceId);
+            case PUT:
+                return this.call(request.put(requestBody).build(), traceId);
+            case DELETE:
+                return this.call(request.delete(requestBody).build(), traceId);
+            default:
+                return new Result().error("请求方式错误!");
+        }
+    }
+
+    private Result call(Request request, String traceId) {
+        OkHttpClient client = HttpClientBuilder.getClient();
+
+        try (Response response = client.newCall(request).execute();
+             ResponseBody body = response.body();) {
+            String bodyStr = body.string();
+
+            if (response.isSuccessful()) {
+                return new Result().success(bodyStr);
+            }
+
+            log.warn(String.format("[%s][response] code:%s body:%s", traceId, response.code(), bodyStr));
+            return new Result().error(bodyStr);
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+            if (e instanceof ConnectException) {
+                return new Result().error("服务访问失败!");
+            }
+        }
+
+        return new Result().error("请求异常!");
+    }
+
+    private RequestBody appendBody(String body) {
+        if (StringUtils.isBlank(body)) {
+            body = "";
+        }
+
+        return FormBody.create(MediaType.parse(Constants.CHARSET_JSON_UTF8), body);
+    }
+
+    private String appendParams(Map<String, String> params) {
+        StringBuilder result = new StringBuilder();
+        result.append("?r=").append(random.nextInt(10000));
+
+        if (MapUtils.isNotEmpty(params)) {
+            for (Map.Entry<String, String> entry : params.entrySet()) {
+                if (StringUtils.isBlank(entry.getKey())) {
+                    continue;
+                }
+
+                result.append("&").append(entry.getKey()).append("=").append(strEncode(entry.getValue()));
+            }
+        }
+
+        return result.toString();
+    }
+
+    private String strEncode(String str) {
+        try {
+            if (StringUtils.isNotBlank(str)) {
+                return URLEncoder.encode(str, "UTF-8");
+            }
+        } catch (UnsupportedEncodingException e) {
+            //ignore
+        }
+        return "";
+    }
+
+    private String getRandomTraceId() {
+        return UUID.randomUUID().toString().replace("-", "");
+    }
+
+}

+ 27 - 0
src/main/java/cn/com/qmth/examcloud/app/service/UpYunService.java

@@ -0,0 +1,27 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:39:29.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service;
+
+/**
+ * 又拍云文件服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/31
+ */
+public interface UpYunService {
+
+    /**
+     * 下载已上传的“作答文件”
+     *
+     * @param filePath
+     * @return
+     * @throws Exception
+     */
+    String downloadPaperAnswer(String filePath) throws Exception;
+
+}

+ 326 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreAuthServiceImpl.java

@@ -0,0 +1,326 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:19:06.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.core.SysProperty;
+import cn.com.qmth.examcloud.app.core.exception.ApiException;
+import cn.com.qmth.examcloud.app.core.utils.HttpClientBuilder;
+import cn.com.qmth.examcloud.app.core.utils.HttpUtils;
+import cn.com.qmth.examcloud.app.core.utils.JsonMapper;
+import cn.com.qmth.examcloud.app.core.utils.ThreadUtils;
+import cn.com.qmth.examcloud.app.model.*;
+import cn.com.qmth.examcloud.app.service.CoreAuthService;
+import cn.com.qmth.examcloud.app.service.RedisService;
+import cn.com.qmth.examcloud.core.basic.api.StudentCloudService;
+import cn.com.qmth.examcloud.core.basic.api.request.GetStudentReq;
+import cn.com.qmth.examcloud.core.basic.api.response.GetStudentResp;
+import cn.com.qmth.examcloud.support.cache.CacheHelper;
+import cn.com.qmth.examcloud.support.cache.bean.OrgPropertyCacheBean;
+import okhttp3.*;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+import org.springframework.util.Assert;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import static cn.com.qmth.examcloud.app.model.Constants.*;
+
+/**
+ * 认证中心业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class CoreAuthServiceImpl implements CoreAuthService {
+
+    private static Logger log = LoggerFactory.getLogger(CoreAuthServiceImpl.class);
+
+    @Autowired
+    private SysProperty sysProperty;
+
+    @Autowired
+    private RedisService redisService;
+
+    @Autowired
+    private StudentCloudService studentCloudService;
+
+    @Override
+    public boolean isDoingExam(Long rootOrgId, String accountType, String account) {
+        if (rootOrgId == null) {
+            log.warn("[Check Doing Exam] rootOrgId is null");
+            return false;
+        }
+
+        log.info(String.format("[Check Doing Exam] rootOrgId=%s, account=%s, accountType=%s", rootOrgId, account, accountType));
+
+        Map<String, String> params = new HashMap<>();
+        params.put("rootOrgId", rootOrgId.toString());
+        if (LoginType.STUDENT_CODE.name().equals(accountType)) {
+            params.put("studentCode", account);
+        } else if (LoginType.STUDENT_IDENTITY_NUMBER.name().equals(accountType)) {
+            params.put("identityNumber", account);
+        } else {
+            //LoginType.STUDENT_PHONE
+            params.put("phone", account);
+        }
+
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/getStudentOnLineExamInfo", sysProperty.getCoreOeStudentUrl());
+        try {
+            //执行请求
+            String json = new JsonMapper().toJson(params);
+            RequestBody formBody = FormBody.create(MediaType.parse(Constants.CHARSET_JSON_UTF8), json);
+            Request request = new Request.Builder()
+                    .post(formBody)
+                    .url(requestUrl)
+                    .build();
+            Response response = HttpClientBuilder.getClient().newCall(request).execute();
+
+            String bodyStr = response.body().string();
+            if (response.isSuccessful()) {
+                ExamInfo info = new JsonMapper().fromJson(bodyStr, ExamInfo.class);
+                if (info != null) {
+                    return info.getExistExamingRecord();
+                }
+            } else {
+                log.warn(bodyStr);
+            }
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+        }
+
+        log.warn("[Check Doing Exam] fail");
+        return false;
+    }
+
+    @Override
+    public boolean isOpenApp(Long rootOrgId) {
+        if (rootOrgId == null) {
+            log.warn("[Check Open APP] rootOrgId is null");
+            return false;
+        }
+
+        try {
+            OrgPropertyCacheBean property = CacheHelper.getOrgProperty(rootOrgId, "APP_ENABLED");
+
+            if ("true".equals(property.getValue())) {
+                return true;
+            }
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+        }
+        return false;
+    }
+
+    @Override
+    public Long getRootOrgIdBySecurityPhone(String securityPhone) {
+        GetStudentReq req = new GetStudentReq();
+        req.setSecurityPhone(securityPhone);
+
+        try {
+            GetStudentResp resp = studentCloudService.getStudent(req);
+            return resp.getStudentInfo().getRootOrgId();
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+        }
+
+        return null;
+    }
+
+    @Override
+    public Result<UserInfo> login(LoginInfo loginInfo) {
+        Assert.notNull(loginInfo, "LoginInfo must be not null.");
+        if (StringUtils.isBlank(loginInfo.getAccountType())) {
+            loginInfo.setAccountType(LoginType.STUDENT_PHONE.name());
+        }
+
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/auth/login", sysProperty.getCoreBasicUrl());
+        Map<String, String> params = new HashMap<>();
+        params.put("accountValue", loginInfo.getAccount());
+        params.put("password", loginInfo.getPassword());
+        params.put("accountType", loginInfo.getAccountType());
+        params.put("rootOrgId", loginInfo.getRootOrgId() != null ? loginInfo.getRootOrgId().toString() : "");
+        params.put("domain", loginInfo.getDomain());
+        params.put("smsCode", loginInfo.getSmsCode());
+        if (loginInfo.getNoSession() != null && loginInfo.getNoSession()) {
+            params.put("noSession", loginInfo.getNoSession().toString());
+        }
+        String json = new JsonMapper().toJson(params);
+
+        RequestBody formBody = FormBody.create(MediaType.parse(Constants.CHARSET_JSON_UTF8), json);
+        Request request = new Request.Builder()
+                .url(requestUrl)
+                .post(formBody)
+                .build();
+
+        //执行请求
+        try (Response response = HttpClientBuilder.getClient().newCall(request).execute();) {
+            String bodyStr = response.body().string();
+            if (response.isSuccessful()) {
+                //获取用户信息
+                UserInfo userInfo = new JsonMapper().fromJson(bodyStr, UserInfo.class);
+                return new Result().success(userInfo);
+            } else {
+                log.warn("Http response is " + bodyStr);
+                ResBody body = new JsonMapper().fromJson(bodyStr, ResBody.class);
+                if (body != null && body.getCode() != null) {
+                    return new Result().error(body.getDesc());
+                }
+                return new Result().error(bodyStr);
+            }
+        } catch (Exception e) {
+            log.error(e.getMessage(), e);
+            return new Result().error("登录异常!");
+        }
+    }
+
+    @Override
+    public Result logout(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/auth/logout", sysProperty.getCoreAuthUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add(PARAM_KEY, key)
+                .add(PARAM_TOKEN, token)
+                .build();
+
+        //清除缓存用户登录信息
+        this.removeCacheLoginInfo(key);
+
+        //退出登录
+        return HttpUtils.doPost(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result updateStudentPassword(String key, String token, Long studentId, String password, String newPassword) throws Exception {
+        Assert.notNull(studentId, "StudentId must be not null.");
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/student/password", sysProperty.getCoreAuthUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add("studentId", studentId.toString())
+                .add("password", password)
+                .add("newPassword", newPassword)
+                .build();
+        return HttpUtils.doPut(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result resetStudentPassword(String key, String token, String newPassword) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/student/password/direct", sysProperty.getCoreAuthUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add("newPassword", newPassword)
+                .build();
+        return HttpUtils.doPut(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result sendSmsCode(String key, String token, String phone) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/auth/sendVerificationCode", sysProperty.getCoreAuthUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add("phone", phone)
+                .build();
+        return HttpUtils.doPost(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result code4Student(String phone, Boolean bound) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/auth/sendVerificationCode4Student", sysProperty.getCoreAuthUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add("phone", phone)
+                .add("bound", bound.toString())
+                .build();
+        Request request = new Request.Builder()
+                .url(requestUrl)
+                .post(formBody)
+                .build();
+
+        //执行请求
+        String traceId = HttpUtils.getRandomTraceId();
+        log.info("[TRACE_ID] " + traceId + " [POST] " + requestUrl);
+        return HttpUtils.call(request, traceId);
+    }
+
+    @Override
+    public Result userBindingPhone(String key, String token, String phone, String code) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/auth/bindSecurityPhone", sysProperty.getCoreAuthUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add("phone", phone)
+                .add("verificationCode", code)
+                .build();
+        return HttpUtils.doPost(requestUrl, formBody, key, token);
+    }
+
+    /**
+     * 缓存用户登录信息
+     */
+    @Override
+    public void cacheLoginInfo(LoginInfo loginInfo, String key) {
+        if (StringUtils.isEmpty(key)) {
+            throw new ApiException("key must be not empty.");
+        }
+        String jsonStr = new JsonMapper().toJson(loginInfo);
+        redisService.set(APP_SESSION_USER_KEY_PREFIX + key, jsonStr, APP_SESSION_EXPIRE_TIME);
+    }
+
+    /**
+     * 清除缓存用户登录信息
+     */
+    private void removeCacheLoginInfo(String key) {
+        if (StringUtils.isEmpty(key)) {
+            throw new ApiException("key must be not empty.");
+        }
+        redisService.delete(APP_SESSION_USER_KEY_PREFIX + key);
+    }
+
+    /**
+     * 获取缓存中的用户登录信息
+     */
+    @Override
+    public LoginInfo getLoginInfo(String key) {
+        if (StringUtils.isEmpty(key)) {
+            throw new ApiException("key must be not empty.");
+        }
+        return redisService.fromJson(APP_SESSION_USER_KEY_PREFIX + key, LoginInfo.class);
+    }
+
+    /**
+     * 获取平台端的默认过期时间(秒)
+     */
+    @Override
+    public int getSessionTimeout() {
+        try {
+            String timeout = redisService.get(Constants.PLATFORM_SESSION_TIMEOUT_KEY);
+            if (StringUtils.isNotEmpty(timeout)) {
+                return Integer.parseInt(timeout);
+            }
+        } catch (Exception e) {
+            //ignore
+        }
+        return Constants.PLATFORM_SESSION_EXPIRE_TIME;
+    }
+
+    /**
+     * 初始化内部接口请求鉴权
+     */
+    @Override
+    public void initRequestTrace() {
+        String key = "C_" + ThreadUtils.getTraceID();
+        Long millis = System.currentTimeMillis();
+        redisService.set(key, millis.toString(), 15);
+    }
+
+}

+ 39 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreBasicServiceImpl.java

@@ -0,0 +1,39 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-08-02 15:16:01.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.core.utils.HttpUtils;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreBasicService;
+import cn.com.qmth.examcloud.app.core.SysProperty;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+/**
+ * 基础信息服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class CoreBasicServiceImpl implements CoreBasicService {
+    private static Logger log = LoggerFactory.getLogger(CoreBasicServiceImpl.class);
+    @Autowired
+    private SysProperty sysProperty;
+
+    @Override
+    public Result getStudentInfo(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_core/student/getStudentInfoBySession", sysProperty.getCoreBasicUrl());
+        //包含头像、手机号等字段
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+}

+ 96 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreExamWorkServiceImpl.java

@@ -0,0 +1,96 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:19:06.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.core.utils.HttpUtils;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreExamWorkService;
+import cn.com.qmth.examcloud.app.core.SysProperty;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+/**
+ * 考务业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class CoreExamWorkServiceImpl implements CoreExamWorkService {
+    private static Logger log = LoggerFactory.getLogger(CoreExamWorkServiceImpl.class);
+    @Autowired
+    private SysProperty sysProperty;
+
+    @Override
+    public Result getPracticeExamList(String key, String token, String studentId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_exam_work/exam/queryByNameLike?name=&examTypes=PRACTICE&studentId=%s", sysProperty.getCoreExamWorkUrl(), studentId);
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            String data = result.getData();
+            //替换掉图片标签内容
+            data = HttpUtils.replaceImages(data);
+            //过滤掉为空的属性
+            data = HttpUtils.filterNullAttributes(data);
+            result.setData(data);
+        }
+        return result;
+    }
+
+    @Override
+    public Result getExamInfo(String key, String token, Long examId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_exam_work/exam/%s", sysProperty.getCoreExamWorkUrl(), examId);
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getBeforeExamRemark(String key, String token, Long examId, String type) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_exam_work/exam/examOrgProperty/%s/%s", sysProperty.getCoreExamWorkUrl(), examId, type);
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getFreezeTime(String key, String token, Long examId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_exam_work/exam/examOrgProperty/%s/FREEZE_TIME", sysProperty.getCoreExamWorkUrl(), examId);
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getUpLoadType(String key, String token, Long examId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_exam_work/exam/examOrgProperty/%s/OFFLINE_UPLOAD_FILE_TYPE", sysProperty.getCoreExamWorkUrl(), examId);
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+}

+ 308 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreOeServiceImpl.java

@@ -0,0 +1,308 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:19:06.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.core.exception.ApiException;
+import cn.com.qmth.examcloud.app.core.utils.DateUtils;
+import cn.com.qmth.examcloud.app.core.utils.HttpUtils;
+import cn.com.qmth.examcloud.app.core.utils.JsonMapper;
+import cn.com.qmth.examcloud.app.model.Constants;
+import cn.com.qmth.examcloud.app.model.GetYunSignatureReq;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.CoreOeService;
+import cn.com.qmth.examcloud.app.core.SysProperty;
+import okhttp3.FormBody;
+import okhttp3.FormBody.Builder;
+import okhttp3.MediaType;
+import okhttp3.MultipartBody;
+import okhttp3.RequestBody;
+import org.apache.commons.io.FilenameUtils;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+import org.springframework.util.Assert;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import static cn.com.qmth.examcloud.app.model.Constants.FILE_TYPE_PDF;
+import static cn.com.qmth.examcloud.app.model.Constants.FILE_TYPE_ZIP;
+
+/**
+ * 网考业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class CoreOeServiceImpl implements CoreOeService {
+    private static Logger log = LoggerFactory.getLogger(CoreOeServiceImpl.class);
+    @Autowired
+    private SysProperty sysProperty;
+
+    @Override
+    public Result getCurrentTime(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/currentTime", sysProperty.getCoreOeStudentUrl());
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //转换日期格式
+            result.setData(DateUtils.formatLongDate(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getOfflineExamCourseList(String key, String token) throws Exception {
+        //封装请求参数									
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/offlineExam/getOfflineCourse", sysProperty.getCoreOeAdminUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result startOfflineExamRecord(String key, String token, String examStudentId) throws Exception {
+        //封装请求参数									
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/offlineExam/startOfflineExam?examStudentId=%s", sysProperty.getCoreOeAdminUrl(), examStudentId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result uploadPaperAnswer(String key, String token, String examRecordId, byte[] fileBytes, String fileName, String md5) throws Exception {
+        Assert.notNull(examRecordId, "FileName must not be null.");
+        Assert.notNull(fileName, "ExamRecordId must not be null.");
+        String fileType = FilenameUtils.getExtension(fileName.toLowerCase());
+        if (!FILE_TYPE_ZIP.equals(fileType) && !FILE_TYPE_PDF.equals(fileType)) {
+            throw new ApiException("FileType must be zip or pdf.");
+        }
+        if (fileBytes.length == 0) {
+            throw new ApiException("File must be not empty.");
+        }
+        //封装请求参数									
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/offlineExam/submitPaper?examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordId);
+        MultipartBody.Builder form = new MultipartBody.Builder().setType(MultipartBody.FORM);
+        RequestBody body = RequestBody.create(MediaType.parse("application/octet-stream"), fileBytes);
+        form.addFormDataPart("file", fileName, body);
+        form.addFormDataPart("fileType", fileType);
+        form.addFormDataPart("md5", md5 != null ? md5 : "");
+        return HttpUtils.doPost(requestUrl, form.build(), key, token);
+    }
+
+    @Override
+    public Result getPracticeExamCourseList(String key, String token, String examId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/practice/queryPracticeCourseList?examId=%s", sysProperty.getCoreOeAdminUrl(), examId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordHeartbeat(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/examHeartbeat", sysProperty.getCoreOeStudentUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result startPracticeExamRecord(String key, String token, String examStudentId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/startExam?examStudentId=%s", sysProperty.getCoreOeStudentUrl(), examStudentId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getAdminExamPaperStructList(String key, String token, String examRecordId,String fromCache) throws Exception {
+        //封装请求参数
+        String requestUrl=null;
+        if("1".equals(fromCache)) {
+            requestUrl = String.format("%s/api/ecs_oe_admin/examRecordPaperStruct/getExamRecordPaperStruct?fromCache=1&examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordId);
+        }else {
+            requestUrl = String.format("%s/api/ecs_oe_admin/examRecordPaperStruct/getExamRecordPaperStruct?examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordId);
+        }
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+    @Override
+    public Result getStudentExamPaperStructList(String key, String token, String examRecordId) throws Exception {
+        //封装请求参数
+        String requestUrl = String.format("%s/api/ecs_oe_student/examRecordPaperStruct/getExamRecordPaperStruct?fromCache=1&examRecordDataId=%s", sysProperty.getCoreOeStudentUrl(), examRecordId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordPaperQuestionList(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examQuestion/findExamQuestionList", sysProperty.getCoreOeStudentUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordPaperQuestionDetail(String key, String token, String questionId, String examRecordId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examQuestion/getQuestionContent?questionId=%s&exam_record_id=%s", sysProperty.getCoreOeStudentUrl(), questionId, examRecordId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result updateExamRecordPaperQuestionAnswer(String key, String token, Integer order, String studentAnswer) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examQuestion/submitQuestionAnswer", sysProperty.getCoreOeStudentUrl());
+        List<Map> paramsList = new ArrayList<Map>();
+        Map<String, Object> params = new HashMap<>();
+        params.put("order", order);
+        params.put("studentAnswer", studentAnswer);
+        paramsList.add(params);
+        String json = new JsonMapper().toJson(paramsList);
+        RequestBody formBody = FormBody.create(MediaType.parse(Constants.CHARSET_JSON_UTF8), json);
+        return HttpUtils.doPost(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result submitPracticeExamRecord(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/endExam", sysProperty.getCoreOeStudentUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result checkOnlineExamRecord(String key, String token) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/checkExamInProgress", sysProperty.getCoreOeStudentUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordPracticeHistoryList(String key, String token, String examStudentId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/practice/queryPracticeRecordList?examStudentId=%s", sysProperty.getCoreOeAdminUrl(), examStudentId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordTotalInfo(String key, String token, String examRecordId,String fromCache) throws Exception {
+        //封装请求参数
+        String requestUrl=null;
+        if("1".equals(fromCache)) {
+            requestUrl = String.format("%s/api/ecs_oe_admin/practice/getPracticeDetailInfo?fromCache=1&examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordId);
+        }else {
+            requestUrl = String.format("%s/api/ecs_oe_admin/practice/getPracticeDetailInfo?examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordId);
+        }
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordQuestionDetailList(String key, String token, String examRecordId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe/report/exam_detail_report/paper_question?exam_record_id=%s", sysProperty.getCoreOeStudentUrl(), examRecordId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getExamRecordQuestionAudioPlayTimes(String key, String token, String questionId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/exam_question_playtimes?questionId=%s", sysProperty.getCoreOeStudentUrl(), questionId);
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result updateExamRecordQuestionAudioPlayTimes(String key, String token, String questionId, String mediaName) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/exam_question_playtimes", sysProperty.getCoreOeStudentUrl());
+        RequestBody formBody = new FormBody.Builder()
+                .add("questionId", questionId)
+                .add("mediaName", mediaName)
+                .build();
+        return HttpUtils.doPost(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result findExamRecordDataEntity(String key, String token, Long examRecordDataId,String fromCache) throws Exception {
+        //封装请求参数
+        String requestUrl=null;
+        if("1".equals(fromCache)) {
+            requestUrl = String.format("%s/api/ecs_oe_admin/exam/record/data/findExamRecordDataEntity?fromCache=1&examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordDataId);
+        }else {
+            requestUrl = String.format("%s/api/ecs_oe_admin/exam/record/data/findExamRecordDataEntity?examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordDataId);
+        }
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getExamRecordQuestions(String key, String token, Long examRecordDataId,String fromCache) throws Exception {
+        //封装请求参数
+        String requestUrl=null;
+        if("1".equals(fromCache)) {
+            requestUrl = String.format("%s/api/ecs_oe_admin/examRecordQuestions/getExamRecordQuestions?fromCache=1&examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordDataId);
+        }else {
+            requestUrl = String.format("%s/api/ecs_oe_admin/examRecordQuestions/getExamRecordQuestions?examRecordDataId=%s", sysProperty.getCoreOeAdminUrl(), examRecordDataId);
+        }
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getExamPropertyFromCacheByStudentSession(String key, String token,Long examId,String keys) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_exam_work/exam/getExamPropertyFromCacheByStudentSession/"+examId+"/"+keys, sysProperty.getCoreExamWorkUrl());
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        return result;
+    }
+
+    @Override
+    public Result getPaperById(String key, String token, String paperId) throws Exception {
+        final String requestUrl = String.format("%s/api/ecs_ques/paper/"+paperId, sysProperty.getCoreQuestionUrl());
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        return result;
+    }
+    
+    @Override
+    public Result getYunSignature(String key, String token, GetYunSignatureReq req) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/yunSignature", sysProperty.getCoreOeStudentUrl());
+        Builder b= new FormBody.Builder()
+                .add("examRecordDataId", req.getExamRecordDataId())
+                .add("order", req.getOrder())
+                .add("fileMd5", req.getFileMd5())
+                .add("fileSuffix", req.getFileSuffix());
+        if(StringUtils.isNotBlank(req.getExt())) {
+                b.add("ext", req.getExt());
+        }
+        RequestBody formBody =b.build();
+                 
+        return HttpUtils.doPost(requestUrl, formBody, key, token);
+    }
+
+    @Override
+    public Result queryExamList(String key, String token) throws Exception {
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/examControl/queryExamList", sysProperty.getCoreOeAdminUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result getEndExamInfo(String key, String token, Long examRecordDataId) throws Exception {
+        final String requestUrl = String.format("%s/api/ecs_oe_student/examControl/getEndExamInfo?examRecordDataId="+examRecordDataId, sysProperty.getCoreOeStudentUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+    @Override
+    public Result queryObjectiveScoreList(String key, String token, Long examStudentId) throws Exception {
+        final String requestUrl = String.format("%s/api/ecs_oe_admin/exam/score/queryObjectiveScoreList?examStudentId="+examStudentId, sysProperty.getCoreOeAdminUrl());
+        return HttpUtils.doGet(requestUrl, key, token);
+    }
+
+}

+ 92 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/CoreQuestionServiceImpl.java

@@ -0,0 +1,92 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:19:06.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.core.utils.HttpUtils;
+import cn.com.qmth.examcloud.app.core.utils.JsonMapper;
+import cn.com.qmth.examcloud.app.model.Constants;
+import cn.com.qmth.examcloud.app.model.ResBody;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.core.SysProperty;
+import cn.com.qmth.examcloud.app.service.CoreQuestionService;
+import okhttp3.FormBody;
+import okhttp3.MediaType;
+import okhttp3.RequestBody;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+import java.net.URLEncoder;
+import java.util.HashMap;
+import java.util.Map;
+
+/**
+ * 题库业务服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class CoreQuestionServiceImpl implements CoreQuestionService {
+    private static Logger log = LoggerFactory.getLogger(CoreQuestionServiceImpl.class);
+    @Autowired
+    private SysProperty sysProperty;
+
+    @Override
+    public String downloadPaper(String key, String token, String paperId, String orgName) throws Exception {
+        //封装请求参数
+        orgName = URLEncoder.encode(orgName, "utf-8");
+        return String.format("%s/api/ecs_ques/paper/export/%s/PAPER/%s/%s/offLine?$key=%s&$token=%s", sysProperty.getCoreQuestionUrl(), paperId, orgName, paperId, key, token);
+    }
+
+    @Override
+    public Result getPaperDetail(String key, String token, String paperId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_ques/paper/%s", sysProperty.getCoreQuestionUrl(), paperId);
+        Result<String> result = HttpUtils.doGet(requestUrl, key, token);
+        if (result.isSuccess()) {
+            //过滤掉为空的属性
+            result.setData(HttpUtils.filterNullAttributes(result.getData()));
+        }
+        return result;
+    }
+
+    @Override
+    public Result getQuestion(String key, String token, String courseCode, Long examId, String groupCode, String questionId) throws Exception {
+        //封装请求参数
+        final String requestUrl = String.format("%s/api/ecs_ques/default_question/question", sysProperty.getCoreQuestionUrl());
+        Map<String, Object> params = new HashMap<>();
+        params.put("courseCode", courseCode);
+        params.put("examId", examId);
+        params.put("groupCode", groupCode);
+        params.put("questionId", questionId);
+        String json = new JsonMapper().toJson(params);
+        RequestBody formBody = FormBody.create(MediaType.parse(Constants.CHARSET_JSON_UTF8), json);
+        //发送短信
+        Result<String> result = HttpUtils.doPost(requestUrl, formBody, key, token);
+        return this.parseResult(result);
+    }
+
+    private Result<String> parseResult(Result<String> result) {
+        if (!result.isSuccess()) {
+            return result;
+        }
+        ResBody body = new JsonMapper().fromJson(result.getData(), ResBody.class);
+        if (body != null && body.getSuccess() != null) {
+            if (body.getSuccess() == true) {
+                return new Result().success();
+            } else {
+                return new Result().error(body.getReturnMsg());
+            }
+        }
+        return result;
+    }
+
+}

+ 107 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/DeviceRecordServiceImpl.java

@@ -0,0 +1,107 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:14:51.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.dao.DeviceRecordRepository;
+import cn.com.qmth.examcloud.app.model.DeviceRecord;
+import cn.com.qmth.examcloud.app.model.DeviceRecordQuery;
+import cn.com.qmth.examcloud.app.model.Result;
+import cn.com.qmth.examcloud.app.service.DeviceRecordService;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.data.domain.Page;
+import org.springframework.data.domain.PageRequest;
+import org.springframework.data.domain.Pageable;
+import org.springframework.data.domain.Sort;
+import org.springframework.data.jpa.domain.Specification;
+import org.springframework.scheduling.annotation.Async;
+import org.springframework.stereotype.Service;
+import org.springframework.util.Assert;
+
+import javax.persistence.criteria.Predicate;
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * 设备访问记录Service
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class DeviceRecordServiceImpl implements DeviceRecordService {
+    private static Logger log = LoggerFactory.getLogger(DeviceRecordServiceImpl.class);
+    @Autowired
+    private DeviceRecordRepository deviceRecordRepository;
+
+    @SuppressWarnings("unchecked")
+    @Override
+    public Result<Page<DeviceRecord>> getDeviceRecordList(DeviceRecordQuery params) throws Exception {
+        Specification<DeviceRecord> spec = (Specification<DeviceRecord>) (root, query, builder) -> {
+            List<Predicate> predicates = new ArrayList<>();
+            if (StringUtils.isNotBlank(params.getSystem())) {
+                predicates.add(builder.equal(root.get("system"), params.getSystem().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getDeviceId())) {
+                predicates.add(builder.equal(root.get("deviceId"), params.getDeviceId().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getNetType())) {
+                predicates.add(builder.equal(root.get("netType"), params.getNetType().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getBrand())) {
+                predicates.add(builder.equal(root.get("brand"), params.getBrand().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getModel())) {
+                predicates.add(builder.equal(root.get("model"), params.getModel().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getSysVersion())) {
+                predicates.add(builder.equal(root.get("sysVersion"), params.getSysVersion().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getAppVersion())) {
+                predicates.add(builder.equal(root.get("appVersion"), params.getAppVersion().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getAppCode())) {
+                predicates.add(builder.equal(root.get("appCode"), params.getAppCode().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getPatchCode())) {
+                predicates.add(builder.equal(root.get("patchCode"), params.getPatchCode().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getAccount())) {
+                predicates.add(builder.equal(root.get("account"), params.getAccount().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getLoginKey())) {
+                predicates.add(builder.equal(root.get("loginKey"), params.getLoginKey().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getLoginToken())) {
+                predicates.add(builder.equal(root.get("loginToken"), params.getLoginToken().trim()));
+            }
+            if (StringUtils.isNotBlank(params.getUrl())) {
+                predicates.add(builder.like(root.get("url"), "%" + params.getUrl().trim() + "%"));
+            }
+            if (StringUtils.isNotBlank(params.getIp())) {
+                predicates.add(builder.equal(root.get("ip"), params.getIp().trim()));
+            }
+            return builder.and(predicates.toArray(new Predicate[predicates.size()]));
+        };
+
+        Sort sort = new Sort(Sort.Direction.DESC, "id");
+        Pageable pageable = PageRequest.of(params.getPageNo() - 1, params.getPageSize(), sort);
+        Page<DeviceRecord> page = deviceRecordRepository.findAll(spec, pageable);
+        return new Result<>().success(page);
+    }
+
+    @Async
+    @Override
+    public void addDeviceRecord(DeviceRecord deviceRecord) throws Exception {
+        Assert.notNull(deviceRecord, "DeviceRecord must not be null.");
+        deviceRecordRepository.save(deviceRecord);
+    }
+
+}

+ 35 - 0
src/main/java/cn/com/qmth/examcloud/app/service/impl/UpYunServiceImpl.java

@@ -0,0 +1,35 @@
+/*
+ * *************************************************
+ * Copyright (c) 2018 QMTH. All Rights Reserved.
+ * Created by Deason on 2018-07-31 17:19:06.
+ * *************************************************
+ */
+
+package cn.com.qmth.examcloud.app.service.impl;
+
+import cn.com.qmth.examcloud.app.core.SysProperty;
+import cn.com.qmth.examcloud.app.service.UpYunService;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+/**
+ * 又拍云文件服务接口
+ *
+ * @author: fengdesheng
+ * @since: 2018/7/16
+ */
+@Service
+public class UpYunServiceImpl implements UpYunService {
+    private static Logger log = LoggerFactory.getLogger(UpYunServiceImpl.class);
+    @Autowired
+    private SysProperty sysProperty;
+
+    @Override
+    public String downloadPaperAnswer(String filePath) throws Exception {
+        //封装请求参数
+        return String.format("%s/%s", sysProperty.getDomain(), filePath);
+    }
+
+}

+ 6 - 0
src/main/resources/application.properties

@@ -0,0 +1,6 @@
+spring.profiles.active=test
+
+examcloud.startup.startupCode=8090
+examcloud.startup.configCenterHost=config-host
+examcloud.startup.configCenterPort=9999
+examcloud.startup.appCode=API

+ 1 - 0
src/main/resources/classpath.location

@@ -0,0 +1 @@
+classpath 定位文件

+ 82 - 0
src/main/resources/log4j2.xml

@@ -0,0 +1,82 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<Configuration status="WARN" monitorInterval="30">
+
+    <Properties>
+        <Property name="commonLevel" value="${sys:log.commonLevel}"/>
+    </Properties>
+
+    <Appenders>
+        <!-- 控制台 日志 -->
+        <Console name="Console" target="SYSTEM_OUT">
+            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS}| %level | %X{TRACE_ID} - %X{CALLER} | %m | %l%n"/>
+        </Console>
+        <!-- debug 日志 -->
+        <RollingFile name="DEBUG_APPENDER" fileName="./logs/debug/debug.log"
+                     filePattern="./logs/debug/debug-%d{yyyy.MM.dd.HH}-%i.log">
+            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS}| %level | %X{TRACE_ID} - %X{CALLER} | %m | %l%n"/>
+            <Policies>
+                <TimeBasedTriggeringPolicy interval="1"/>
+                <SizeBasedTriggeringPolicy size="100 MB"/>
+            </Policies>
+            <DefaultRolloverStrategy max="10000">
+                <Delete basePath="./logs/debug" maxDepth="1">
+                    <IfFileName glob="debug-*.log">
+                        <IfAccumulatedFileSize exceeds="2 GB"/>
+                    </IfFileName>
+                </Delete>
+            </DefaultRolloverStrategy>
+        </RollingFile>
+        <!-- 接口日志 -->
+        <RollingFile name="INTERFACE_APPENDER" fileName="./logs/interface/interface.log"
+                     filePattern="./logs/interface/interface-%d{yyyy.MM.dd.HH}-%i.log">
+            <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss.SSS}| %level | %X{TRACE_ID} - %X{CALLER} | %m%n"/>
+            <Policies>
+                <TimeBasedTriggeringPolicy interval="1"/>
+                <SizeBasedTriggeringPolicy size="100 MB"/>
+            </Policies>
+            <DefaultRolloverStrategy max="10000">
+                <Delete basePath="./logs/interface" maxDepth="1">
+                    <IfFileName glob="interface-*.log">
+                        <IfAccumulatedFileSize exceeds="10 GB"/>
+                    </IfFileName>
+                </Delete>
+            </DefaultRolloverStrategy>
+        </RollingFile>
+    </Appenders>
+
+    <Loggers>
+        <Logger name="cn.com.qmth" level="${commonLevel}" additivity="false">
+            <AppenderRef ref="DEBUG_APPENDER"/>
+            <AppenderRef ref="Console"/>
+        </Logger>
+
+        <Logger name="INTERFACE_LOGGER" level="INFO" additivity="false">
+            <AppenderRef ref="INTERFACE_APPENDER"/>
+            <AppenderRef ref="Console"/>
+        </Logger>
+
+        <Logger name="org.hibernate.type.descriptor.sql.BasicBinder" level="${commonLevel}"/>
+        <Logger name="org.hibernate.type.descriptor.sql.BasicExtractor" level="${commonLevel}"/>
+        <Logger name="org.hibernate.SQL" level="${commonLevel}"/>
+        <Logger name="org.hibernate.type" level="${commonLevel}"/>
+        <Logger name="org.hibernate.engine.QueryParameters" level="${commonLevel}"/>
+        <Logger name="org.hibernate.engine.query.HQLQueryPlan" level="${commonLevel}"/>
+
+        <logger name="org.springframework" level="ERROR"/>
+        <logger name="org.springframework.jdbc.core.JdbcTemplate" level="WARN"/>
+        <logger name="org.springframework.data.mongodb" level="WARN"/>
+        <logger name="org.docx4j" level="ERROR"/>
+        <logger name="org.docx4j.model.properties.run.FontSize" level="OFF"/>
+        <logger name="com.netflix" level="ERROR"/>
+        <logger name="com.netflix.discovery" level="ERROR"/>
+        <logger name="org.apache" level="ERROR"/>
+        <logger name="springfox.documentation" level="ERROR"/>
+        <logger name="cn.afterturn.easypoi" level="ERROR"/>
+
+        <Root level="INFO">
+            <AppenderRef ref="Console"/>
+            <AppenderRef ref="DEBUG_APPENDER"/>
+        </Root>
+    </Loggers>
+
+</Configuration>

+ 5 - 0
src/main/resources/security-exclusions.conf

@@ -0,0 +1,5 @@
+regexp:.*swagger.*
+[][/doc.html][GET]
+[][/][GET]
+[${$rmp}/device/record][/list][GET]
+[${$rmp}/device/record][/list][POST]

+ 0 - 0
src/main/resources/security.properties


+ 99 - 0
src/main/resources/static/api/app-api/page.js

@@ -0,0 +1,99 @@
+(function ($) {
+    var ms = {
+        init: function (obj, args) {
+            obj.unbind();
+            return (function () {
+                ms.fillHtml(obj, args);
+                ms.bindEvent(obj, args);
+            })();
+        },
+        fillHtml: function (obj, args) {
+            return (function () {
+                obj.empty();
+                if (args.totalPages > 0) {
+                    /* 上一页 */
+                    if (args.current > 1) {
+                        obj.append('<li><a href="#" curId="prevPage">&laquo;</a></li>');
+                    } else {
+                        obj.append('<li class="disabled"><a>&laquo;</a></li>');
+                    }
+                    /* 中间页 */
+                    if (args.current != 1 && args.current >= 4 && args.totalPages != 4) {
+                        obj.append('<li><a href="#" curId="curNumber">1</a></li>');
+                    }
+                    if (args.current - 2 > 2 && args.current <= args.totalPages && args.totalPages > 5) {
+                        obj.append('<li><a>…</a></li>');
+                    }
+                    var start = args.current - 2, end = args.current + 2;
+                    if ((start > 1 && args.current < 4) || args.current == 1) {
+                        end++;
+                    }
+                    if (args.current > args.totalPages - 4 && args.current >= args.totalPages) {
+                        start--;
+                    }
+                    for (; start <= end; start++) {
+                        if (start <= args.totalPages && start >= 1) {
+                            if (start != args.current) {
+                                obj.append('<li><a href="#" curId="curNumber">' + start + '</a></li>');
+                            } else {
+                                obj.append('<li class="active"><a>' + start + '</a></li>');
+                            }
+                        }
+                    }
+                    if (args.current + 2 < args.totalPages - 1 && args.current >= 1 && args.totalPages > 5) {
+                        obj.append('<li><a>…</a></li>');
+                    }
+                    if (args.current != args.totalPages && args.current < args.totalPages - 2 && args.totalPages != 4) {
+                        obj.append('<li><a href="#" curId="curNumber">' + args.totalPages + '</a></li>');
+                    }
+                    /* 下一页 */
+                    if (args.current < args.totalPages) {
+                        obj.append('<li><a href="#" curId="nextPage">&raquo;</a></li>');
+                    } else {
+                        obj.append('<li class="disabled"><a>&raquo;</a></li>');
+                    }
+                }
+                obj.append('<li class="disabled"><a>共 ' + args.totalElements + ' 条,共 ' + args.totalPages + ' 页</a></li>');
+            })();
+        },
+        /* 绑定事件 */
+        bindEvent: function (obj, args) {
+            return (function () {
+                obj.on("click", "a[curId=curNumber]", function () {
+                    var current = parseInt($(this).text());
+                    ms.fillHtml(obj, {"current": current, "totalPages": args.totalPages});
+                    if (typeof(args.backFn) == "function") {
+                        args.backFn(current);
+                    }
+                });
+                /* 上一页 */
+                obj.on("click", "a[curId=prevPage]", function () {
+                    var current = parseInt(obj.children("li.active").text());
+                    ms.fillHtml(obj, {"current": current - 1, "totalPages": args.totalPages});
+                    if (typeof(args.backFn) == "function") {
+                        args.backFn(current - 1);
+                    }
+                });
+                /* 下一页 */
+                obj.on("click", "a[curId=nextPage]", function () {
+                    var current = parseInt(obj.children("li.active").text());
+                    ms.fillHtml(obj, {"current": current + 1, "totalPages": args.totalPages});
+                    if (typeof(args.backFn) == "function") {
+                        args.backFn(current + 1);
+                    }
+                });
+            })();
+        }
+    };
+
+    $.fn.createPage = function (options) {
+        var args = $.extend({
+            totalElements: 0,
+            totalPages: 0,
+            current: 1,
+            backFn: function () {
+            }
+        }, options);
+        ms.init(this, args);
+    };
+})(jQuery);

Fichier diff supprimé car celui-ci est trop grand
+ 4 - 0
src/main/resources/static/api/app-api/styles/bootstrap/css/bootstrap.min.css


+ 0 - 0
src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.eot


+ 288 - 0
src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.svg

@@ -0,0 +1,288 @@
+<?xml version="1.0" standalone="no"?>
+<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" >
+<svg xmlns="http://www.w3.org/2000/svg">
+<metadata></metadata>
+<defs>
+<font id="glyphicons_halflingsregular" horiz-adv-x="1200" >
+<font-face units-per-em="1200" ascent="960" descent="-240" />
+<missing-glyph horiz-adv-x="500" />
+<glyph horiz-adv-x="0" />
+<glyph horiz-adv-x="400" />
+<glyph unicode=" " />
+<glyph unicode="*" d="M600 1100q15 0 34 -1.5t30 -3.5l11 -1q10 -2 17.5 -10.5t7.5 -18.5v-224l158 158q7 7 18 8t19 -6l106 -106q7 -8 6 -19t-8 -18l-158 -158h224q10 0 18.5 -7.5t10.5 -17.5q6 -41 6 -75q0 -15 -1.5 -34t-3.5 -30l-1 -11q-2 -10 -10.5 -17.5t-18.5 -7.5h-224l158 -158 q7 -7 8 -18t-6 -19l-106 -106q-8 -7 -19 -6t-18 8l-158 158v-224q0 -10 -7.5 -18.5t-17.5 -10.5q-41 -6 -75 -6q-15 0 -34 1.5t-30 3.5l-11 1q-10 2 -17.5 10.5t-7.5 18.5v224l-158 -158q-7 -7 -18 -8t-19 6l-106 106q-7 8 -6 19t8 18l158 158h-224q-10 0 -18.5 7.5 t-10.5 17.5q-6 41 -6 75q0 15 1.5 34t3.5 30l1 11q2 10 10.5 17.5t18.5 7.5h224l-158 158q-7 7 -8 18t6 19l106 106q8 7 19 6t18 -8l158 -158v224q0 10 7.5 18.5t17.5 10.5q41 6 75 6z" />
+<glyph unicode="+" d="M450 1100h200q21 0 35.5 -14.5t14.5 -35.5v-350h350q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-350v-350q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v350h-350q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5 h350v350q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xa0;" />
+<glyph unicode="&#xa5;" d="M825 1100h250q10 0 12.5 -5t-5.5 -13l-364 -364q-6 -6 -11 -18h268q10 0 13 -6t-3 -14l-120 -160q-6 -8 -18 -14t-22 -6h-125v-100h275q10 0 13 -6t-3 -14l-120 -160q-6 -8 -18 -14t-22 -6h-125v-174q0 -11 -7.5 -18.5t-18.5 -7.5h-148q-11 0 -18.5 7.5t-7.5 18.5v174 h-275q-10 0 -13 6t3 14l120 160q6 8 18 14t22 6h125v100h-275q-10 0 -13 6t3 14l120 160q6 8 18 14t22 6h118q-5 12 -11 18l-364 364q-8 8 -5.5 13t12.5 5h250q25 0 43 -18l164 -164q8 -8 18 -8t18 8l164 164q18 18 43 18z" />
+<glyph unicode="&#x2000;" horiz-adv-x="650" />
+<glyph unicode="&#x2001;" horiz-adv-x="1300" />
+<glyph unicode="&#x2002;" horiz-adv-x="650" />
+<glyph unicode="&#x2003;" horiz-adv-x="1300" />
+<glyph unicode="&#x2004;" horiz-adv-x="433" />
+<glyph unicode="&#x2005;" horiz-adv-x="325" />
+<glyph unicode="&#x2006;" horiz-adv-x="216" />
+<glyph unicode="&#x2007;" horiz-adv-x="216" />
+<glyph unicode="&#x2008;" horiz-adv-x="162" />
+<glyph unicode="&#x2009;" horiz-adv-x="260" />
+<glyph unicode="&#x200a;" horiz-adv-x="72" />
+<glyph unicode="&#x202f;" horiz-adv-x="260" />
+<glyph unicode="&#x205f;" horiz-adv-x="325" />
+<glyph unicode="&#x20ac;" d="M744 1198q242 0 354 -189q60 -104 66 -209h-181q0 45 -17.5 82.5t-43.5 61.5t-58 40.5t-60.5 24t-51.5 7.5q-19 0 -40.5 -5.5t-49.5 -20.5t-53 -38t-49 -62.5t-39 -89.5h379l-100 -100h-300q-6 -50 -6 -100h406l-100 -100h-300q9 -74 33 -132t52.5 -91t61.5 -54.5t59 -29 t47 -7.5q22 0 50.5 7.5t60.5 24.5t58 41t43.5 61t17.5 80h174q-30 -171 -128 -278q-107 -117 -274 -117q-206 0 -324 158q-36 48 -69 133t-45 204h-217l100 100h112q1 47 6 100h-218l100 100h134q20 87 51 153.5t62 103.5q117 141 297 141z" />
+<glyph unicode="&#x20bd;" d="M428 1200h350q67 0 120 -13t86 -31t57 -49.5t35 -56.5t17 -64.5t6.5 -60.5t0.5 -57v-16.5v-16.5q0 -36 -0.5 -57t-6.5 -61t-17 -65t-35 -57t-57 -50.5t-86 -31.5t-120 -13h-178l-2 -100h288q10 0 13 -6t-3 -14l-120 -160q-6 -8 -18 -14t-22 -6h-138v-175q0 -11 -5.5 -18 t-15.5 -7h-149q-10 0 -17.5 7.5t-7.5 17.5v175h-267q-10 0 -13 6t3 14l120 160q6 8 18 14t22 6h117v100h-267q-10 0 -13 6t3 14l120 160q6 8 18 14t22 6h117v475q0 10 7.5 17.5t17.5 7.5zM600 1000v-300h203q64 0 86.5 33t22.5 119q0 84 -22.5 116t-86.5 32h-203z" />
+<glyph unicode="&#x2212;" d="M250 700h800q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#x231b;" d="M1000 1200v-150q0 -21 -14.5 -35.5t-35.5 -14.5h-50v-100q0 -91 -49.5 -165.5t-130.5 -109.5q81 -35 130.5 -109.5t49.5 -165.5v-150h50q21 0 35.5 -14.5t14.5 -35.5v-150h-800v150q0 21 14.5 35.5t35.5 14.5h50v150q0 91 49.5 165.5t130.5 109.5q-81 35 -130.5 109.5 t-49.5 165.5v100h-50q-21 0 -35.5 14.5t-14.5 35.5v150h800zM400 1000v-100q0 -60 32.5 -109.5t87.5 -73.5q28 -12 44 -37t16 -55t-16 -55t-44 -37q-55 -24 -87.5 -73.5t-32.5 -109.5v-150h400v150q0 60 -32.5 109.5t-87.5 73.5q-28 12 -44 37t-16 55t16 55t44 37 q55 24 87.5 73.5t32.5 109.5v100h-400z" />
+<glyph unicode="&#x25fc;" horiz-adv-x="500" d="M0 0z" />
+<glyph unicode="&#x2601;" d="M503 1089q110 0 200.5 -59.5t134.5 -156.5q44 14 90 14q120 0 205 -86.5t85 -206.5q0 -121 -85 -207.5t-205 -86.5h-750q-79 0 -135.5 57t-56.5 137q0 69 42.5 122.5t108.5 67.5q-2 12 -2 37q0 153 108 260.5t260 107.5z" />
+<glyph unicode="&#x26fa;" d="M774 1193.5q16 -9.5 20.5 -27t-5.5 -33.5l-136 -187l467 -746h30q20 0 35 -18.5t15 -39.5v-42h-1200v42q0 21 15 39.5t35 18.5h30l468 746l-135 183q-10 16 -5.5 34t20.5 28t34 5.5t28 -20.5l111 -148l112 150q9 16 27 20.5t34 -5zM600 200h377l-182 112l-195 534v-646z " />
+<glyph unicode="&#x2709;" d="M25 1100h1150q10 0 12.5 -5t-5.5 -13l-564 -567q-8 -8 -18 -8t-18 8l-564 567q-8 8 -5.5 13t12.5 5zM18 882l264 -264q8 -8 8 -18t-8 -18l-264 -264q-8 -8 -13 -5.5t-5 12.5v550q0 10 5 12.5t13 -5.5zM918 618l264 264q8 8 13 5.5t5 -12.5v-550q0 -10 -5 -12.5t-13 5.5 l-264 264q-8 8 -8 18t8 18zM818 482l364 -364q8 -8 5.5 -13t-12.5 -5h-1150q-10 0 -12.5 5t5.5 13l364 364q8 8 18 8t18 -8l164 -164q8 -8 18 -8t18 8l164 164q8 8 18 8t18 -8z" />
+<glyph unicode="&#x270f;" d="M1011 1210q19 0 33 -13l153 -153q13 -14 13 -33t-13 -33l-99 -92l-214 214l95 96q13 14 32 14zM1013 800l-615 -614l-214 214l614 614zM317 96l-333 -112l110 335z" />
+<glyph unicode="&#xe001;" d="M700 650v-550h250q21 0 35.5 -14.5t14.5 -35.5v-50h-800v50q0 21 14.5 35.5t35.5 14.5h250v550l-500 550h1200z" />
+<glyph unicode="&#xe002;" d="M368 1017l645 163q39 15 63 0t24 -49v-831q0 -55 -41.5 -95.5t-111.5 -63.5q-79 -25 -147 -4.5t-86 75t25.5 111.5t122.5 82q72 24 138 8v521l-600 -155v-606q0 -42 -44 -90t-109 -69q-79 -26 -147 -5.5t-86 75.5t25.5 111.5t122.5 82.5q72 24 138 7v639q0 38 14.5 59 t53.5 34z" />
+<glyph unicode="&#xe003;" d="M500 1191q100 0 191 -39t156.5 -104.5t104.5 -156.5t39 -191l-1 -2l1 -5q0 -141 -78 -262l275 -274q23 -26 22.5 -44.5t-22.5 -42.5l-59 -58q-26 -20 -46.5 -20t-39.5 20l-275 274q-119 -77 -261 -77l-5 1l-2 -1q-100 0 -191 39t-156.5 104.5t-104.5 156.5t-39 191 t39 191t104.5 156.5t156.5 104.5t191 39zM500 1022q-88 0 -162 -43t-117 -117t-43 -162t43 -162t117 -117t162 -43t162 43t117 117t43 162t-43 162t-117 117t-162 43z" />
+<glyph unicode="&#xe005;" d="M649 949q48 68 109.5 104t121.5 38.5t118.5 -20t102.5 -64t71 -100.5t27 -123q0 -57 -33.5 -117.5t-94 -124.5t-126.5 -127.5t-150 -152.5t-146 -174q-62 85 -145.5 174t-150 152.5t-126.5 127.5t-93.5 124.5t-33.5 117.5q0 64 28 123t73 100.5t104 64t119 20 t120.5 -38.5t104.5 -104z" />
+<glyph unicode="&#xe006;" d="M407 800l131 353q7 19 17.5 19t17.5 -19l129 -353h421q21 0 24 -8.5t-14 -20.5l-342 -249l130 -401q7 -20 -0.5 -25.5t-24.5 6.5l-343 246l-342 -247q-17 -12 -24.5 -6.5t-0.5 25.5l130 400l-347 251q-17 12 -14 20.5t23 8.5h429z" />
+<glyph unicode="&#xe007;" d="M407 800l131 353q7 19 17.5 19t17.5 -19l129 -353h421q21 0 24 -8.5t-14 -20.5l-342 -249l130 -401q7 -20 -0.5 -25.5t-24.5 6.5l-343 246l-342 -247q-17 -12 -24.5 -6.5t-0.5 25.5l130 400l-347 251q-17 12 -14 20.5t23 8.5h429zM477 700h-240l197 -142l-74 -226 l193 139l195 -140l-74 229l192 140h-234l-78 211z" />
+<glyph unicode="&#xe008;" d="M600 1200q124 0 212 -88t88 -212v-250q0 -46 -31 -98t-69 -52v-75q0 -10 6 -21.5t15 -17.5l358 -230q9 -5 15 -16.5t6 -21.5v-93q0 -10 -7.5 -17.5t-17.5 -7.5h-1150q-10 0 -17.5 7.5t-7.5 17.5v93q0 10 6 21.5t15 16.5l358 230q9 6 15 17.5t6 21.5v75q-38 0 -69 52 t-31 98v250q0 124 88 212t212 88z" />
+<glyph unicode="&#xe009;" d="M25 1100h1150q10 0 17.5 -7.5t7.5 -17.5v-1050q0 -10 -7.5 -17.5t-17.5 -7.5h-1150q-10 0 -17.5 7.5t-7.5 17.5v1050q0 10 7.5 17.5t17.5 7.5zM100 1000v-100h100v100h-100zM875 1000h-550q-10 0 -17.5 -7.5t-7.5 -17.5v-350q0 -10 7.5 -17.5t17.5 -7.5h550 q10 0 17.5 7.5t7.5 17.5v350q0 10 -7.5 17.5t-17.5 7.5zM1000 1000v-100h100v100h-100zM100 800v-100h100v100h-100zM1000 800v-100h100v100h-100zM100 600v-100h100v100h-100zM1000 600v-100h100v100h-100zM875 500h-550q-10 0 -17.5 -7.5t-7.5 -17.5v-350q0 -10 7.5 -17.5 t17.5 -7.5h550q10 0 17.5 7.5t7.5 17.5v350q0 10 -7.5 17.5t-17.5 7.5zM100 400v-100h100v100h-100zM1000 400v-100h100v100h-100zM100 200v-100h100v100h-100zM1000 200v-100h100v100h-100z" />
+<glyph unicode="&#xe010;" d="M50 1100h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM650 1100h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400 q0 21 14.5 35.5t35.5 14.5zM50 500h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM650 500h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400 q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe011;" d="M50 1100h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM450 1100h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200 q0 21 14.5 35.5t35.5 14.5zM850 1100h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM50 700h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200 q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM450 700h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM850 700h200q21 0 35.5 -14.5t14.5 -35.5v-200 q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM50 300h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM450 300h200 q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM850 300h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5 t35.5 14.5z" />
+<glyph unicode="&#xe012;" d="M50 1100h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM450 1100h700q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5v200 q0 21 14.5 35.5t35.5 14.5zM50 700h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM450 700h700q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700 q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM50 300h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5zM450 300h700q21 0 35.5 -14.5t14.5 -35.5v-200 q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe013;" d="M465 477l571 571q8 8 18 8t17 -8l177 -177q8 -7 8 -17t-8 -18l-783 -784q-7 -8 -17.5 -8t-17.5 8l-384 384q-8 8 -8 18t8 17l177 177q7 8 17 8t18 -8l171 -171q7 -7 18 -7t18 7z" />
+<glyph unicode="&#xe014;" d="M904 1083l178 -179q8 -8 8 -18.5t-8 -17.5l-267 -268l267 -268q8 -7 8 -17.5t-8 -18.5l-178 -178q-8 -8 -18.5 -8t-17.5 8l-268 267l-268 -267q-7 -8 -17.5 -8t-18.5 8l-178 178q-8 8 -8 18.5t8 17.5l267 268l-267 268q-8 7 -8 17.5t8 18.5l178 178q8 8 18.5 8t17.5 -8 l268 -267l268 268q7 7 17.5 7t18.5 -7z" />
+<glyph unicode="&#xe015;" d="M507 1177q98 0 187.5 -38.5t154.5 -103.5t103.5 -154.5t38.5 -187.5q0 -141 -78 -262l300 -299q8 -8 8 -18.5t-8 -18.5l-109 -108q-7 -8 -17.5 -8t-18.5 8l-300 299q-119 -77 -261 -77q-98 0 -188 38.5t-154.5 103t-103 154.5t-38.5 188t38.5 187.5t103 154.5 t154.5 103.5t188 38.5zM506.5 1023q-89.5 0 -165.5 -44t-120 -120.5t-44 -166t44 -165.5t120 -120t165.5 -44t166 44t120.5 120t44 165.5t-44 166t-120.5 120.5t-166 44zM425 900h150q10 0 17.5 -7.5t7.5 -17.5v-75h75q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5 t-17.5 -7.5h-75v-75q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v75h-75q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5h75v75q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe016;" d="M507 1177q98 0 187.5 -38.5t154.5 -103.5t103.5 -154.5t38.5 -187.5q0 -141 -78 -262l300 -299q8 -8 8 -18.5t-8 -18.5l-109 -108q-7 -8 -17.5 -8t-18.5 8l-300 299q-119 -77 -261 -77q-98 0 -188 38.5t-154.5 103t-103 154.5t-38.5 188t38.5 187.5t103 154.5 t154.5 103.5t188 38.5zM506.5 1023q-89.5 0 -165.5 -44t-120 -120.5t-44 -166t44 -165.5t120 -120t165.5 -44t166 44t120.5 120t44 165.5t-44 166t-120.5 120.5t-166 44zM325 800h350q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-350q-10 0 -17.5 7.5 t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe017;" d="M550 1200h100q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM800 975v166q167 -62 272 -209.5t105 -331.5q0 -117 -45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5 t-184.5 123t-123 184.5t-45.5 224q0 184 105 331.5t272 209.5v-166q-103 -55 -165 -155t-62 -220q0 -116 57 -214.5t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5q0 120 -62 220t-165 155z" />
+<glyph unicode="&#xe018;" d="M1025 1200h150q10 0 17.5 -7.5t7.5 -17.5v-1150q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v1150q0 10 7.5 17.5t17.5 7.5zM725 800h150q10 0 17.5 -7.5t7.5 -17.5v-750q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v750 q0 10 7.5 17.5t17.5 7.5zM425 500h150q10 0 17.5 -7.5t7.5 -17.5v-450q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v450q0 10 7.5 17.5t17.5 7.5zM125 300h150q10 0 17.5 -7.5t7.5 -17.5v-250q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5 v250q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe019;" d="M600 1174q33 0 74 -5l38 -152l5 -1q49 -14 94 -39l5 -2l134 80q61 -48 104 -105l-80 -134l3 -5q25 -44 39 -93l1 -6l152 -38q5 -43 5 -73q0 -34 -5 -74l-152 -38l-1 -6q-15 -49 -39 -93l-3 -5l80 -134q-48 -61 -104 -105l-134 81l-5 -3q-44 -25 -94 -39l-5 -2l-38 -151 q-43 -5 -74 -5q-33 0 -74 5l-38 151l-5 2q-49 14 -94 39l-5 3l-134 -81q-60 48 -104 105l80 134l-3 5q-25 45 -38 93l-2 6l-151 38q-6 42 -6 74q0 33 6 73l151 38l2 6q13 48 38 93l3 5l-80 134q47 61 105 105l133 -80l5 2q45 25 94 39l5 1l38 152q43 5 74 5zM600 815 q-89 0 -152 -63t-63 -151.5t63 -151.5t152 -63t152 63t63 151.5t-63 151.5t-152 63z" />
+<glyph unicode="&#xe020;" d="M500 1300h300q41 0 70.5 -29.5t29.5 -70.5v-100h275q10 0 17.5 -7.5t7.5 -17.5v-75h-1100v75q0 10 7.5 17.5t17.5 7.5h275v100q0 41 29.5 70.5t70.5 29.5zM500 1200v-100h300v100h-300zM1100 900v-800q0 -41 -29.5 -70.5t-70.5 -29.5h-700q-41 0 -70.5 29.5t-29.5 70.5 v800h900zM300 800v-700h100v700h-100zM500 800v-700h100v700h-100zM700 800v-700h100v700h-100zM900 800v-700h100v700h-100z" />
+<glyph unicode="&#xe021;" d="M18 618l620 608q8 7 18.5 7t17.5 -7l608 -608q8 -8 5.5 -13t-12.5 -5h-175v-575q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v375h-300v-375q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v575h-175q-10 0 -12.5 5t5.5 13z" />
+<glyph unicode="&#xe022;" d="M600 1200v-400q0 -41 29.5 -70.5t70.5 -29.5h300v-650q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v1100q0 21 14.5 35.5t35.5 14.5h450zM1000 800h-250q-21 0 -35.5 14.5t-14.5 35.5v250z" />
+<glyph unicode="&#xe023;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5t-57 214.5t-155.5 155.5t-214.5 57zM525 900h50q10 0 17.5 -7.5t7.5 -17.5v-275h175q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v350q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe024;" d="M1300 0h-538l-41 400h-242l-41 -400h-538l431 1200h209l-21 -300h162l-20 300h208zM515 800l-27 -300h224l-27 300h-170z" />
+<glyph unicode="&#xe025;" d="M550 1200h200q21 0 35.5 -14.5t14.5 -35.5v-450h191q20 0 25.5 -11.5t-7.5 -27.5l-327 -400q-13 -16 -32 -16t-32 16l-327 400q-13 16 -7.5 27.5t25.5 11.5h191v450q0 21 14.5 35.5t35.5 14.5zM1125 400h50q10 0 17.5 -7.5t7.5 -17.5v-350q0 -10 -7.5 -17.5t-17.5 -7.5 h-1050q-10 0 -17.5 7.5t-7.5 17.5v350q0 10 7.5 17.5t17.5 7.5h50q10 0 17.5 -7.5t7.5 -17.5v-175h900v175q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe026;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5t-57 214.5t-155.5 155.5t-214.5 57zM525 900h150q10 0 17.5 -7.5t7.5 -17.5v-275h137q21 0 26 -11.5t-8 -27.5l-223 -275q-13 -16 -32 -16t-32 16l-223 275q-13 16 -8 27.5t26 11.5h137v275q0 10 7.5 17.5t17.5 7.5z " />
+<glyph unicode="&#xe027;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5t-57 214.5t-155.5 155.5t-214.5 57zM632 914l223 -275q13 -16 8 -27.5t-26 -11.5h-137v-275q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v275h-137q-21 0 -26 11.5t8 27.5l223 275q13 16 32 16 t32 -16z" />
+<glyph unicode="&#xe028;" d="M225 1200h750q10 0 19.5 -7t12.5 -17l186 -652q7 -24 7 -49v-425q0 -12 -4 -27t-9 -17q-12 -6 -37 -6h-1100q-12 0 -27 4t-17 8q-6 13 -6 38l1 425q0 25 7 49l185 652q3 10 12.5 17t19.5 7zM878 1000h-556q-10 0 -19 -7t-11 -18l-87 -450q-2 -11 4 -18t16 -7h150 q10 0 19.5 -7t11.5 -17l38 -152q2 -10 11.5 -17t19.5 -7h250q10 0 19.5 7t11.5 17l38 152q2 10 11.5 17t19.5 7h150q10 0 16 7t4 18l-87 450q-2 11 -11 18t-19 7z" />
+<glyph unicode="&#xe029;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5t-57 214.5t-155.5 155.5t-214.5 57zM540 820l253 -190q17 -12 17 -30t-17 -30l-253 -190q-16 -12 -28 -6.5t-12 26.5v400q0 21 12 26.5t28 -6.5z" />
+<glyph unicode="&#xe030;" d="M947 1060l135 135q7 7 12.5 5t5.5 -13v-362q0 -10 -7.5 -17.5t-17.5 -7.5h-362q-11 0 -13 5.5t5 12.5l133 133q-109 76 -238 76q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5h150q0 -117 -45.5 -224 t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5q192 0 347 -117z" />
+<glyph unicode="&#xe031;" d="M947 1060l135 135q7 7 12.5 5t5.5 -13v-361q0 -11 -7.5 -18.5t-18.5 -7.5h-361q-11 0 -13 5.5t5 12.5l134 134q-110 75 -239 75q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5h-150q0 117 45.5 224t123 184.5t184.5 123t224 45.5q192 0 347 -117zM1027 600h150 q0 -117 -45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5q-192 0 -348 118l-134 -134q-7 -8 -12.5 -5.5t-5.5 12.5v360q0 11 7.5 18.5t18.5 7.5h360q10 0 12.5 -5.5t-5.5 -12.5l-133 -133q110 -76 240 -76q116 0 214.5 57t155.5 155.5t57 214.5z" />
+<glyph unicode="&#xe032;" d="M125 1200h1050q10 0 17.5 -7.5t7.5 -17.5v-1150q0 -10 -7.5 -17.5t-17.5 -7.5h-1050q-10 0 -17.5 7.5t-7.5 17.5v1150q0 10 7.5 17.5t17.5 7.5zM1075 1000h-850q-10 0 -17.5 -7.5t-7.5 -17.5v-850q0 -10 7.5 -17.5t17.5 -7.5h850q10 0 17.5 7.5t7.5 17.5v850 q0 10 -7.5 17.5t-17.5 7.5zM325 900h50q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-50q-10 0 -17.5 7.5t-7.5 17.5v50q0 10 7.5 17.5t17.5 7.5zM525 900h450q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-450q-10 0 -17.5 7.5t-7.5 17.5v50 q0 10 7.5 17.5t17.5 7.5zM325 700h50q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-50q-10 0 -17.5 7.5t-7.5 17.5v50q0 10 7.5 17.5t17.5 7.5zM525 700h450q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-450q-10 0 -17.5 7.5t-7.5 17.5v50 q0 10 7.5 17.5t17.5 7.5zM325 500h50q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-50q-10 0 -17.5 7.5t-7.5 17.5v50q0 10 7.5 17.5t17.5 7.5zM525 500h450q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-450q-10 0 -17.5 7.5t-7.5 17.5v50 q0 10 7.5 17.5t17.5 7.5zM325 300h50q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-50q-10 0 -17.5 7.5t-7.5 17.5v50q0 10 7.5 17.5t17.5 7.5zM525 300h450q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-450q-10 0 -17.5 7.5t-7.5 17.5v50 q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe033;" d="M900 800v200q0 83 -58.5 141.5t-141.5 58.5h-300q-82 0 -141 -59t-59 -141v-200h-100q-41 0 -70.5 -29.5t-29.5 -70.5v-600q0 -41 29.5 -70.5t70.5 -29.5h900q41 0 70.5 29.5t29.5 70.5v600q0 41 -29.5 70.5t-70.5 29.5h-100zM400 800v150q0 21 15 35.5t35 14.5h200 q20 0 35 -14.5t15 -35.5v-150h-300z" />
+<glyph unicode="&#xe034;" d="M125 1100h50q10 0 17.5 -7.5t7.5 -17.5v-1075h-100v1075q0 10 7.5 17.5t17.5 7.5zM1075 1052q4 0 9 -2q16 -6 16 -23v-421q0 -6 -3 -12q-33 -59 -66.5 -99t-65.5 -58t-56.5 -24.5t-52.5 -6.5q-26 0 -57.5 6.5t-52.5 13.5t-60 21q-41 15 -63 22.5t-57.5 15t-65.5 7.5 q-85 0 -160 -57q-7 -5 -15 -5q-6 0 -11 3q-14 7 -14 22v438q22 55 82 98.5t119 46.5q23 2 43 0.5t43 -7t32.5 -8.5t38 -13t32.5 -11q41 -14 63.5 -21t57 -14t63.5 -7q103 0 183 87q7 8 18 8z" />
+<glyph unicode="&#xe035;" d="M600 1175q116 0 227 -49.5t192.5 -131t131 -192.5t49.5 -227v-300q0 -10 -7.5 -17.5t-17.5 -7.5h-50q-10 0 -17.5 7.5t-7.5 17.5v300q0 127 -70.5 231.5t-184.5 161.5t-245 57t-245 -57t-184.5 -161.5t-70.5 -231.5v-300q0 -10 -7.5 -17.5t-17.5 -7.5h-50 q-10 0 -17.5 7.5t-7.5 17.5v300q0 116 49.5 227t131 192.5t192.5 131t227 49.5zM220 500h160q8 0 14 -6t6 -14v-460q0 -8 -6 -14t-14 -6h-160q-8 0 -14 6t-6 14v460q0 8 6 14t14 6zM820 500h160q8 0 14 -6t6 -14v-460q0 -8 -6 -14t-14 -6h-160q-8 0 -14 6t-6 14v460 q0 8 6 14t14 6z" />
+<glyph unicode="&#xe036;" d="M321 814l258 172q9 6 15 2.5t6 -13.5v-750q0 -10 -6 -13.5t-15 2.5l-258 172q-21 14 -46 14h-250q-10 0 -17.5 7.5t-7.5 17.5v350q0 10 7.5 17.5t17.5 7.5h250q25 0 46 14zM900 668l120 120q7 7 17 7t17 -7l34 -34q7 -7 7 -17t-7 -17l-120 -120l120 -120q7 -7 7 -17 t-7 -17l-34 -34q-7 -7 -17 -7t-17 7l-120 119l-120 -119q-7 -7 -17 -7t-17 7l-34 34q-7 7 -7 17t7 17l119 120l-119 120q-7 7 -7 17t7 17l34 34q7 8 17 8t17 -8z" />
+<glyph unicode="&#xe037;" d="M321 814l258 172q9 6 15 2.5t6 -13.5v-750q0 -10 -6 -13.5t-15 2.5l-258 172q-21 14 -46 14h-250q-10 0 -17.5 7.5t-7.5 17.5v350q0 10 7.5 17.5t17.5 7.5h250q25 0 46 14zM766 900h4q10 -1 16 -10q96 -129 96 -290q0 -154 -90 -281q-6 -9 -17 -10l-3 -1q-9 0 -16 6 l-29 23q-7 7 -8.5 16.5t4.5 17.5q72 103 72 229q0 132 -78 238q-6 8 -4.5 18t9.5 17l29 22q7 5 15 5z" />
+<glyph unicode="&#xe038;" d="M967 1004h3q11 -1 17 -10q135 -179 135 -396q0 -105 -34 -206.5t-98 -185.5q-7 -9 -17 -10h-3q-9 0 -16 6l-42 34q-8 6 -9 16t5 18q111 150 111 328q0 90 -29.5 176t-84.5 157q-6 9 -5 19t10 16l42 33q7 5 15 5zM321 814l258 172q9 6 15 2.5t6 -13.5v-750q0 -10 -6 -13.5 t-15 2.5l-258 172q-21 14 -46 14h-250q-10 0 -17.5 7.5t-7.5 17.5v350q0 10 7.5 17.5t17.5 7.5h250q25 0 46 14zM766 900h4q10 -1 16 -10q96 -129 96 -290q0 -154 -90 -281q-6 -9 -17 -10l-3 -1q-9 0 -16 6l-29 23q-7 7 -8.5 16.5t4.5 17.5q72 103 72 229q0 132 -78 238 q-6 8 -4.5 18.5t9.5 16.5l29 22q7 5 15 5z" />
+<glyph unicode="&#xe039;" d="M500 900h100v-100h-100v-100h-400v-100h-100v600h500v-300zM1200 700h-200v-100h200v-200h-300v300h-200v300h-100v200h600v-500zM100 1100v-300h300v300h-300zM800 1100v-300h300v300h-300zM300 900h-100v100h100v-100zM1000 900h-100v100h100v-100zM300 500h200v-500 h-500v500h200v100h100v-100zM800 300h200v-100h-100v-100h-200v100h-100v100h100v200h-200v100h300v-300zM100 400v-300h300v300h-300zM300 200h-100v100h100v-100zM1200 200h-100v100h100v-100zM700 0h-100v100h100v-100zM1200 0h-300v100h300v-100z" />
+<glyph unicode="&#xe040;" d="M100 200h-100v1000h100v-1000zM300 200h-100v1000h100v-1000zM700 200h-200v1000h200v-1000zM900 200h-100v1000h100v-1000zM1200 200h-200v1000h200v-1000zM400 0h-300v100h300v-100zM600 0h-100v91h100v-91zM800 0h-100v91h100v-91zM1100 0h-200v91h200v-91z" />
+<glyph unicode="&#xe041;" d="M500 1200l682 -682q8 -8 8 -18t-8 -18l-464 -464q-8 -8 -18 -8t-18 8l-682 682l1 475q0 10 7.5 17.5t17.5 7.5h474zM319.5 1024.5q-29.5 29.5 -71 29.5t-71 -29.5t-29.5 -71.5t29.5 -71.5t71 -29.5t71 29.5t29.5 71.5t-29.5 71.5z" />
+<glyph unicode="&#xe042;" d="M500 1200l682 -682q8 -8 8 -18t-8 -18l-464 -464q-8 -8 -18 -8t-18 8l-682 682l1 475q0 10 7.5 17.5t17.5 7.5h474zM800 1200l682 -682q8 -8 8 -18t-8 -18l-464 -464q-8 -8 -18 -8t-18 8l-56 56l424 426l-700 700h150zM319.5 1024.5q-29.5 29.5 -71 29.5t-71 -29.5 t-29.5 -71.5t29.5 -71.5t71 -29.5t71 29.5t29.5 71.5t-29.5 71.5z" />
+<glyph unicode="&#xe043;" d="M300 1200h825q75 0 75 -75v-900q0 -25 -18 -43l-64 -64q-8 -8 -13 -5.5t-5 12.5v950q0 10 -7.5 17.5t-17.5 7.5h-700q-25 0 -43 -18l-64 -64q-8 -8 -5.5 -13t12.5 -5h700q10 0 17.5 -7.5t7.5 -17.5v-950q0 -10 -7.5 -17.5t-17.5 -7.5h-850q-10 0 -17.5 7.5t-7.5 17.5v975 q0 25 18 43l139 139q18 18 43 18z" />
+<glyph unicode="&#xe044;" d="M250 1200h800q21 0 35.5 -14.5t14.5 -35.5v-1150l-450 444l-450 -445v1151q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe045;" d="M822 1200h-444q-11 0 -19 -7.5t-9 -17.5l-78 -301q-7 -24 7 -45l57 -108q6 -9 17.5 -15t21.5 -6h450q10 0 21.5 6t17.5 15l62 108q14 21 7 45l-83 301q-1 10 -9 17.5t-19 7.5zM1175 800h-150q-10 0 -21 -6.5t-15 -15.5l-78 -156q-4 -9 -15 -15.5t-21 -6.5h-550 q-10 0 -21 6.5t-15 15.5l-78 156q-4 9 -15 15.5t-21 6.5h-150q-10 0 -17.5 -7.5t-7.5 -17.5v-650q0 -10 7.5 -17.5t17.5 -7.5h150q10 0 17.5 7.5t7.5 17.5v150q0 10 7.5 17.5t17.5 7.5h750q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 7.5 -17.5t17.5 -7.5h150q10 0 17.5 7.5 t7.5 17.5v650q0 10 -7.5 17.5t-17.5 7.5zM850 200h-500q-10 0 -19.5 -7t-11.5 -17l-38 -152q-2 -10 3.5 -17t15.5 -7h600q10 0 15.5 7t3.5 17l-38 152q-2 10 -11.5 17t-19.5 7z" />
+<glyph unicode="&#xe046;" d="M500 1100h200q56 0 102.5 -20.5t72.5 -50t44 -59t25 -50.5l6 -20h150q41 0 70.5 -29.5t29.5 -70.5v-600q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5v600q0 41 29.5 70.5t70.5 29.5h150q2 8 6.5 21.5t24 48t45 61t72 48t102.5 21.5zM900 800v-100 h100v100h-100zM600 730q-95 0 -162.5 -67.5t-67.5 -162.5t67.5 -162.5t162.5 -67.5t162.5 67.5t67.5 162.5t-67.5 162.5t-162.5 67.5zM600 603q43 0 73 -30t30 -73t-30 -73t-73 -30t-73 30t-30 73t30 73t73 30z" />
+<glyph unicode="&#xe047;" d="M681 1199l385 -998q20 -50 60 -92q18 -19 36.5 -29.5t27.5 -11.5l10 -2v-66h-417v66q53 0 75 43.5t5 88.5l-82 222h-391q-58 -145 -92 -234q-11 -34 -6.5 -57t25.5 -37t46 -20t55 -6v-66h-365v66q56 24 84 52q12 12 25 30.5t20 31.5l7 13l399 1006h93zM416 521h340 l-162 457z" />
+<glyph unicode="&#xe048;" d="M753 641q5 -1 14.5 -4.5t36 -15.5t50.5 -26.5t53.5 -40t50.5 -54.5t35.5 -70t14.5 -87q0 -67 -27.5 -125.5t-71.5 -97.5t-98.5 -66.5t-108.5 -40.5t-102 -13h-500v89q41 7 70.5 32.5t29.5 65.5v827q0 24 -0.5 34t-3.5 24t-8.5 19.5t-17 13.5t-28 12.5t-42.5 11.5v71 l471 -1q57 0 115.5 -20.5t108 -57t80.5 -94t31 -124.5q0 -51 -15.5 -96.5t-38 -74.5t-45 -50.5t-38.5 -30.5zM400 700h139q78 0 130.5 48.5t52.5 122.5q0 41 -8.5 70.5t-29.5 55.5t-62.5 39.5t-103.5 13.5h-118v-350zM400 200h216q80 0 121 50.5t41 130.5q0 90 -62.5 154.5 t-156.5 64.5h-159v-400z" />
+<glyph unicode="&#xe049;" d="M877 1200l2 -57q-83 -19 -116 -45.5t-40 -66.5l-132 -839q-9 -49 13 -69t96 -26v-97h-500v97q186 16 200 98l173 832q3 17 3 30t-1.5 22.5t-9 17.5t-13.5 12.5t-21.5 10t-26 8.5t-33.5 10q-13 3 -19 5v57h425z" />
+<glyph unicode="&#xe050;" d="M1300 900h-50q0 21 -4 37t-9.5 26.5t-18 17.5t-22 11t-28.5 5.5t-31 2t-37 0.5h-200v-850q0 -22 25 -34.5t50 -13.5l25 -2v-100h-400v100q4 0 11 0.5t24 3t30 7t24 15t11 24.5v850h-200q-25 0 -37 -0.5t-31 -2t-28.5 -5.5t-22 -11t-18 -17.5t-9.5 -26.5t-4 -37h-50v300 h1000v-300zM175 1000h-75v-800h75l-125 -167l-125 167h75v800h-75l125 167z" />
+<glyph unicode="&#xe051;" d="M1100 900h-50q0 21 -4 37t-9.5 26.5t-18 17.5t-22 11t-28.5 5.5t-31 2t-37 0.5h-200v-650q0 -22 25 -34.5t50 -13.5l25 -2v-100h-400v100q4 0 11 0.5t24 3t30 7t24 15t11 24.5v650h-200q-25 0 -37 -0.5t-31 -2t-28.5 -5.5t-22 -11t-18 -17.5t-9.5 -26.5t-4 -37h-50v300 h1000v-300zM1167 50l-167 -125v75h-800v-75l-167 125l167 125v-75h800v75z" />
+<glyph unicode="&#xe052;" d="M50 1100h600q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-600q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 800h1000q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5v100 q0 21 14.5 35.5t35.5 14.5zM50 500h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 200h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe053;" d="M250 1100h700q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 800h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v100 q0 21 14.5 35.5t35.5 14.5zM250 500h700q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 200h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe054;" d="M500 950v100q0 21 14.5 35.5t35.5 14.5h600q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-600q-21 0 -35.5 14.5t-14.5 35.5zM100 650v100q0 21 14.5 35.5t35.5 14.5h1000q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1000 q-21 0 -35.5 14.5t-14.5 35.5zM300 350v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5zM0 50v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100 q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5z" />
+<glyph unicode="&#xe055;" d="M50 1100h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 800h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v100 q0 21 14.5 35.5t35.5 14.5zM50 500h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 200h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe056;" d="M50 1100h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM350 1100h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v100 q0 21 14.5 35.5t35.5 14.5zM50 800h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM350 800h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-800 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 500h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM350 500h800q21 0 35.5 -14.5t14.5 -35.5v-100 q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 200h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM350 200h800 q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe057;" d="M400 0h-100v1100h100v-1100zM550 1100h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM550 800h500q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-500 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM267 550l-167 -125v75h-200v100h200v75zM550 500h300q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-300q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM550 200h600 q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-600q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe058;" d="M50 1100h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM900 0h-100v1100h100v-1100zM50 800h500q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-500 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM1100 600h200v-100h-200v-75l-167 125l167 125v-75zM50 500h300q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-300q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5zM50 200h600 q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-600q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe059;" d="M75 1000h750q31 0 53 -22t22 -53v-650q0 -31 -22 -53t-53 -22h-750q-31 0 -53 22t-22 53v650q0 31 22 53t53 22zM1200 300l-300 300l300 300v-600z" />
+<glyph unicode="&#xe060;" d="M44 1100h1112q18 0 31 -13t13 -31v-1012q0 -18 -13 -31t-31 -13h-1112q-18 0 -31 13t-13 31v1012q0 18 13 31t31 13zM100 1000v-737l247 182l298 -131l-74 156l293 318l236 -288v500h-1000zM342 884q56 0 95 -39t39 -94.5t-39 -95t-95 -39.5t-95 39.5t-39 95t39 94.5 t95 39z" />
+<glyph unicode="&#xe062;" d="M648 1169q117 0 216 -60t156.5 -161t57.5 -218q0 -115 -70 -258q-69 -109 -158 -225.5t-143 -179.5l-54 -62q-9 8 -25.5 24.5t-63.5 67.5t-91 103t-98.5 128t-95.5 148q-60 132 -60 249q0 88 34 169.5t91.5 142t137 96.5t166.5 36zM652.5 974q-91.5 0 -156.5 -65 t-65 -157t65 -156.5t156.5 -64.5t156.5 64.5t65 156.5t-65 157t-156.5 65z" />
+<glyph unicode="&#xe063;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 173v854q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57z" />
+<glyph unicode="&#xe064;" d="M554 1295q21 -72 57.5 -143.5t76 -130t83 -118t82.5 -117t70 -116t49.5 -126t18.5 -136.5q0 -71 -25.5 -135t-68.5 -111t-99 -82t-118.5 -54t-125.5 -23q-84 5 -161.5 34t-139.5 78.5t-99 125t-37 164.5q0 69 18 136.5t49.5 126.5t69.5 116.5t81.5 117.5t83.5 119 t76.5 131t58.5 143zM344 710q-23 -33 -43.5 -70.5t-40.5 -102.5t-17 -123q1 -37 14.5 -69.5t30 -52t41 -37t38.5 -24.5t33 -15q21 -7 32 -1t13 22l6 34q2 10 -2.5 22t-13.5 19q-5 4 -14 12t-29.5 40.5t-32.5 73.5q-26 89 6 271q2 11 -6 11q-8 1 -15 -10z" />
+<glyph unicode="&#xe065;" d="M1000 1013l108 115q2 1 5 2t13 2t20.5 -1t25 -9.5t28.5 -21.5q22 -22 27 -43t0 -32l-6 -10l-108 -115zM350 1100h400q50 0 105 -13l-187 -187h-368q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v182l200 200v-332 q0 -165 -93.5 -257.5t-256.5 -92.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400q0 165 92.5 257.5t257.5 92.5zM1009 803l-362 -362l-161 -50l55 170l355 355z" />
+<glyph unicode="&#xe066;" d="M350 1100h361q-164 -146 -216 -200h-195q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5l200 153v-103q0 -165 -92.5 -257.5t-257.5 -92.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400q0 165 92.5 257.5t257.5 92.5z M824 1073l339 -301q8 -7 8 -17.5t-8 -17.5l-340 -306q-7 -6 -12.5 -4t-6.5 11v203q-26 1 -54.5 0t-78.5 -7.5t-92 -17.5t-86 -35t-70 -57q10 59 33 108t51.5 81.5t65 58.5t68.5 40.5t67 24.5t56 13.5t40 4.5v210q1 10 6.5 12.5t13.5 -4.5z" />
+<glyph unicode="&#xe067;" d="M350 1100h350q60 0 127 -23l-178 -177h-349q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v69l200 200v-219q0 -165 -92.5 -257.5t-257.5 -92.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400q0 165 92.5 257.5t257.5 92.5z M643 639l395 395q7 7 17.5 7t17.5 -7l101 -101q7 -7 7 -17.5t-7 -17.5l-531 -532q-7 -7 -17.5 -7t-17.5 7l-248 248q-7 7 -7 17.5t7 17.5l101 101q7 7 17.5 7t17.5 -7l111 -111q8 -7 18 -7t18 7z" />
+<glyph unicode="&#xe068;" d="M318 918l264 264q8 8 18 8t18 -8l260 -264q7 -8 4.5 -13t-12.5 -5h-170v-200h200v173q0 10 5 12t13 -5l264 -260q8 -7 8 -17.5t-8 -17.5l-264 -265q-8 -7 -13 -5t-5 12v173h-200v-200h170q10 0 12.5 -5t-4.5 -13l-260 -264q-8 -8 -18 -8t-18 8l-264 264q-8 8 -5.5 13 t12.5 5h175v200h-200v-173q0 -10 -5 -12t-13 5l-264 265q-8 7 -8 17.5t8 17.5l264 260q8 7 13 5t5 -12v-173h200v200h-175q-10 0 -12.5 5t5.5 13z" />
+<glyph unicode="&#xe069;" d="M250 1100h100q21 0 35.5 -14.5t14.5 -35.5v-438l464 453q15 14 25.5 10t10.5 -25v-1000q0 -21 -10.5 -25t-25.5 10l-464 453v-438q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v1000q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe070;" d="M50 1100h100q21 0 35.5 -14.5t14.5 -35.5v-438l464 453q15 14 25.5 10t10.5 -25v-438l464 453q15 14 25.5 10t10.5 -25v-1000q0 -21 -10.5 -25t-25.5 10l-464 453v-438q0 -21 -10.5 -25t-25.5 10l-464 453v-438q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5 t-14.5 35.5v1000q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe071;" d="M1200 1050v-1000q0 -21 -10.5 -25t-25.5 10l-464 453v-438q0 -21 -10.5 -25t-25.5 10l-492 480q-15 14 -15 35t15 35l492 480q15 14 25.5 10t10.5 -25v-438l464 453q15 14 25.5 10t10.5 -25z" />
+<glyph unicode="&#xe072;" d="M243 1074l814 -498q18 -11 18 -26t-18 -26l-814 -498q-18 -11 -30.5 -4t-12.5 28v1000q0 21 12.5 28t30.5 -4z" />
+<glyph unicode="&#xe073;" d="M250 1000h200q21 0 35.5 -14.5t14.5 -35.5v-800q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v800q0 21 14.5 35.5t35.5 14.5zM650 1000h200q21 0 35.5 -14.5t14.5 -35.5v-800q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v800 q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe074;" d="M1100 950v-800q0 -21 -14.5 -35.5t-35.5 -14.5h-800q-21 0 -35.5 14.5t-14.5 35.5v800q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5z" />
+<glyph unicode="&#xe075;" d="M500 612v438q0 21 10.5 25t25.5 -10l492 -480q15 -14 15 -35t-15 -35l-492 -480q-15 -14 -25.5 -10t-10.5 25v438l-464 -453q-15 -14 -25.5 -10t-10.5 25v1000q0 21 10.5 25t25.5 -10z" />
+<glyph unicode="&#xe076;" d="M1048 1102l100 1q20 0 35 -14.5t15 -35.5l5 -1000q0 -21 -14.5 -35.5t-35.5 -14.5l-100 -1q-21 0 -35.5 14.5t-14.5 35.5l-2 437l-463 -454q-14 -15 -24.5 -10.5t-10.5 25.5l-2 437l-462 -455q-15 -14 -25.5 -9.5t-10.5 24.5l-5 1000q0 21 10.5 25.5t25.5 -10.5l466 -450 l-2 438q0 20 10.5 24.5t25.5 -9.5l466 -451l-2 438q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe077;" d="M850 1100h100q21 0 35.5 -14.5t14.5 -35.5v-1000q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v438l-464 -453q-15 -14 -25.5 -10t-10.5 25v1000q0 21 10.5 25t25.5 -10l464 -453v438q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe078;" d="M686 1081l501 -540q15 -15 10.5 -26t-26.5 -11h-1042q-22 0 -26.5 11t10.5 26l501 540q15 15 36 15t36 -15zM150 400h1000q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe079;" d="M885 900l-352 -353l352 -353l-197 -198l-552 552l552 550z" />
+<glyph unicode="&#xe080;" d="M1064 547l-551 -551l-198 198l353 353l-353 353l198 198z" />
+<glyph unicode="&#xe081;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM650 900h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-150h-150 q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -21 14.5 -35.5t35.5 -14.5h150v-150q0 -21 14.5 -35.5t35.5 -14.5h100q21 0 35.5 14.5t14.5 35.5v150h150q21 0 35.5 14.5t14.5 35.5v100q0 21 -14.5 35.5t-35.5 14.5h-150v150q0 21 -14.5 35.5t-35.5 14.5z" />
+<glyph unicode="&#xe082;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM850 700h-500q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -21 14.5 -35.5 t35.5 -14.5h500q21 0 35.5 14.5t14.5 35.5v100q0 21 -14.5 35.5t-35.5 14.5z" />
+<glyph unicode="&#xe083;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM741.5 913q-12.5 0 -21.5 -9l-120 -120l-120 120q-9 9 -21.5 9 t-21.5 -9l-141 -141q-9 -9 -9 -21.5t9 -21.5l120 -120l-120 -120q-9 -9 -9 -21.5t9 -21.5l141 -141q9 -9 21.5 -9t21.5 9l120 120l120 -120q9 -9 21.5 -9t21.5 9l141 141q9 9 9 21.5t-9 21.5l-120 120l120 120q9 9 9 21.5t-9 21.5l-141 141q-9 9 -21.5 9z" />
+<glyph unicode="&#xe084;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM546 623l-84 85q-7 7 -17.5 7t-18.5 -7l-139 -139q-7 -8 -7 -18t7 -18 l242 -241q7 -8 17.5 -8t17.5 8l375 375q7 7 7 17.5t-7 18.5l-139 139q-7 7 -17.5 7t-17.5 -7z" />
+<glyph unicode="&#xe085;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM588 941q-29 0 -59 -5.5t-63 -20.5t-58 -38.5t-41.5 -63t-16.5 -89.5 q0 -25 20 -25h131q30 -5 35 11q6 20 20.5 28t45.5 8q20 0 31.5 -10.5t11.5 -28.5q0 -23 -7 -34t-26 -18q-1 0 -13.5 -4t-19.5 -7.5t-20 -10.5t-22 -17t-18.5 -24t-15.5 -35t-8 -46q-1 -8 5.5 -16.5t20.5 -8.5h173q7 0 22 8t35 28t37.5 48t29.5 74t12 100q0 47 -17 83 t-42.5 57t-59.5 34.5t-64 18t-59 4.5zM675 400h-150q-10 0 -17.5 -7.5t-7.5 -17.5v-150q0 -10 7.5 -17.5t17.5 -7.5h150q10 0 17.5 7.5t7.5 17.5v150q0 10 -7.5 17.5t-17.5 7.5z" />
+<glyph unicode="&#xe086;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM675 1000h-150q-10 0 -17.5 -7.5t-7.5 -17.5v-150q0 -10 7.5 -17.5 t17.5 -7.5h150q10 0 17.5 7.5t7.5 17.5v150q0 10 -7.5 17.5t-17.5 7.5zM675 700h-250q-10 0 -17.5 -7.5t-7.5 -17.5v-50q0 -10 7.5 -17.5t17.5 -7.5h75v-200h-75q-10 0 -17.5 -7.5t-7.5 -17.5v-50q0 -10 7.5 -17.5t17.5 -7.5h350q10 0 17.5 7.5t7.5 17.5v50q0 10 -7.5 17.5 t-17.5 7.5h-75v275q0 10 -7.5 17.5t-17.5 7.5z" />
+<glyph unicode="&#xe087;" d="M525 1200h150q10 0 17.5 -7.5t7.5 -17.5v-194q103 -27 178.5 -102.5t102.5 -178.5h194q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-194q-27 -103 -102.5 -178.5t-178.5 -102.5v-194q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v194 q-103 27 -178.5 102.5t-102.5 178.5h-194q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5h194q27 103 102.5 178.5t178.5 102.5v194q0 10 7.5 17.5t17.5 7.5zM700 893v-168q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v168q-68 -23 -119 -74 t-74 -119h168q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-168q23 -68 74 -119t119 -74v168q0 10 7.5 17.5t17.5 7.5h150q10 0 17.5 -7.5t7.5 -17.5v-168q68 23 119 74t74 119h-168q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5h168 q-23 68 -74 119t-119 74z" />
+<glyph unicode="&#xe088;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5t-57 214.5t-155.5 155.5t-214.5 57zM759 823l64 -64q7 -7 7 -17.5t-7 -17.5l-124 -124l124 -124q7 -7 7 -17.5t-7 -17.5l-64 -64q-7 -7 -17.5 -7t-17.5 7l-124 124l-124 -124q-7 -7 -17.5 -7t-17.5 7l-64 64 q-7 7 -7 17.5t7 17.5l124 124l-124 124q-7 7 -7 17.5t7 17.5l64 64q7 7 17.5 7t17.5 -7l124 -124l124 124q7 7 17.5 7t17.5 -7z" />
+<glyph unicode="&#xe089;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5t57 -214.5 t155.5 -155.5t214.5 -57t214.5 57t155.5 155.5t57 214.5t-57 214.5t-155.5 155.5t-214.5 57zM782 788l106 -106q7 -7 7 -17.5t-7 -17.5l-320 -321q-8 -7 -18 -7t-18 7l-202 203q-8 7 -8 17.5t8 17.5l106 106q7 8 17.5 8t17.5 -8l79 -79l197 197q7 7 17.5 7t17.5 -7z" />
+<glyph unicode="&#xe090;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM600 1027q-116 0 -214.5 -57t-155.5 -155.5t-57 -214.5q0 -120 65 -225 l587 587q-105 65 -225 65zM965 819l-584 -584q104 -62 219 -62q116 0 214.5 57t155.5 155.5t57 214.5q0 115 -62 219z" />
+<glyph unicode="&#xe091;" d="M39 582l522 427q16 13 27.5 8t11.5 -26v-291h550q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-550v-291q0 -21 -11.5 -26t-27.5 8l-522 427q-16 13 -16 32t16 32z" />
+<glyph unicode="&#xe092;" d="M639 1009l522 -427q16 -13 16 -32t-16 -32l-522 -427q-16 -13 -27.5 -8t-11.5 26v291h-550q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h550v291q0 21 11.5 26t27.5 -8z" />
+<glyph unicode="&#xe093;" d="M682 1161l427 -522q13 -16 8 -27.5t-26 -11.5h-291v-550q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v550h-291q-21 0 -26 11.5t8 27.5l427 522q13 16 32 16t32 -16z" />
+<glyph unicode="&#xe094;" d="M550 1200h200q21 0 35.5 -14.5t14.5 -35.5v-550h291q21 0 26 -11.5t-8 -27.5l-427 -522q-13 -16 -32 -16t-32 16l-427 522q-13 16 -8 27.5t26 11.5h291v550q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe095;" d="M639 1109l522 -427q16 -13 16 -32t-16 -32l-522 -427q-16 -13 -27.5 -8t-11.5 26v291q-94 -2 -182 -20t-170.5 -52t-147 -92.5t-100.5 -135.5q5 105 27 193.5t67.5 167t113 135t167 91.5t225.5 42v262q0 21 11.5 26t27.5 -8z" />
+<glyph unicode="&#xe096;" d="M850 1200h300q21 0 35.5 -14.5t14.5 -35.5v-300q0 -21 -10.5 -25t-24.5 10l-94 94l-249 -249q-8 -7 -18 -7t-18 7l-106 106q-7 8 -7 18t7 18l249 249l-94 94q-14 14 -10 24.5t25 10.5zM350 0h-300q-21 0 -35.5 14.5t-14.5 35.5v300q0 21 10.5 25t24.5 -10l94 -94l249 249 q8 7 18 7t18 -7l106 -106q7 -8 7 -18t-7 -18l-249 -249l94 -94q14 -14 10 -24.5t-25 -10.5z" />
+<glyph unicode="&#xe097;" d="M1014 1120l106 -106q7 -8 7 -18t-7 -18l-249 -249l94 -94q14 -14 10 -24.5t-25 -10.5h-300q-21 0 -35.5 14.5t-14.5 35.5v300q0 21 10.5 25t24.5 -10l94 -94l249 249q8 7 18 7t18 -7zM250 600h300q21 0 35.5 -14.5t14.5 -35.5v-300q0 -21 -10.5 -25t-24.5 10l-94 94 l-249 -249q-8 -7 -18 -7t-18 7l-106 106q-7 8 -7 18t7 18l249 249l-94 94q-14 14 -10 24.5t25 10.5z" />
+<glyph unicode="&#xe101;" d="M600 1177q117 0 224 -45.5t184.5 -123t123 -184.5t45.5 -224t-45.5 -224t-123 -184.5t-184.5 -123t-224 -45.5t-224 45.5t-184.5 123t-123 184.5t-45.5 224t45.5 224t123 184.5t184.5 123t224 45.5zM704 900h-208q-20 0 -32 -14.5t-8 -34.5l58 -302q4 -20 21.5 -34.5 t37.5 -14.5h54q20 0 37.5 14.5t21.5 34.5l58 302q4 20 -8 34.5t-32 14.5zM675 400h-150q-10 0 -17.5 -7.5t-7.5 -17.5v-150q0 -10 7.5 -17.5t17.5 -7.5h150q10 0 17.5 7.5t7.5 17.5v150q0 10 -7.5 17.5t-17.5 7.5z" />
+<glyph unicode="&#xe102;" d="M260 1200q9 0 19 -2t15 -4l5 -2q22 -10 44 -23l196 -118q21 -13 36 -24q29 -21 37 -12q11 13 49 35l196 118q22 13 45 23q17 7 38 7q23 0 47 -16.5t37 -33.5l13 -16q14 -21 18 -45l25 -123l8 -44q1 -9 8.5 -14.5t17.5 -5.5h61q10 0 17.5 -7.5t7.5 -17.5v-50 q0 -10 -7.5 -17.5t-17.5 -7.5h-50q-10 0 -17.5 -7.5t-7.5 -17.5v-175h-400v300h-200v-300h-400v175q0 10 -7.5 17.5t-17.5 7.5h-50q-10 0 -17.5 7.5t-7.5 17.5v50q0 10 7.5 17.5t17.5 7.5h61q11 0 18 3t7 8q0 4 9 52l25 128q5 25 19 45q2 3 5 7t13.5 15t21.5 19.5t26.5 15.5 t29.5 7zM915 1079l-166 -162q-7 -7 -5 -12t12 -5h219q10 0 15 7t2 17l-51 149q-3 10 -11 12t-15 -6zM463 917l-177 157q-8 7 -16 5t-11 -12l-51 -143q-3 -10 2 -17t15 -7h231q11 0 12.5 5t-5.5 12zM500 0h-375q-10 0 -17.5 7.5t-7.5 17.5v375h400v-400zM1100 400v-375 q0 -10 -7.5 -17.5t-17.5 -7.5h-375v400h400z" />
+<glyph unicode="&#xe103;" d="M1165 1190q8 3 21 -6.5t13 -17.5q-2 -178 -24.5 -323.5t-55.5 -245.5t-87 -174.5t-102.5 -118.5t-118 -68.5t-118.5 -33t-120 -4.5t-105 9.5t-90 16.5q-61 12 -78 11q-4 1 -12.5 0t-34 -14.5t-52.5 -40.5l-153 -153q-26 -24 -37 -14.5t-11 43.5q0 64 42 102q8 8 50.5 45 t66.5 58q19 17 35 47t13 61q-9 55 -10 102.5t7 111t37 130t78 129.5q39 51 80 88t89.5 63.5t94.5 45t113.5 36t129 31t157.5 37t182 47.5zM1116 1098q-8 9 -22.5 -3t-45.5 -50q-38 -47 -119 -103.5t-142 -89.5l-62 -33q-56 -30 -102 -57t-104 -68t-102.5 -80.5t-85.5 -91 t-64 -104.5q-24 -56 -31 -86t2 -32t31.5 17.5t55.5 59.5q25 30 94 75.5t125.5 77.5t147.5 81q70 37 118.5 69t102 79.5t99 111t86.5 148.5q22 50 24 60t-6 19z" />
+<glyph unicode="&#xe104;" d="M653 1231q-39 -67 -54.5 -131t-10.5 -114.5t24.5 -96.5t47.5 -80t63.5 -62.5t68.5 -46.5t65 -30q-4 7 -17.5 35t-18.5 39.5t-17 39.5t-17 43t-13 42t-9.5 44.5t-2 42t4 43t13.5 39t23 38.5q96 -42 165 -107.5t105 -138t52 -156t13 -159t-19 -149.5q-13 -55 -44 -106.5 t-68 -87t-78.5 -64.5t-72.5 -45t-53 -22q-72 -22 -127 -11q-31 6 -13 19q6 3 17 7q13 5 32.5 21t41 44t38.5 63.5t21.5 81.5t-6.5 94.5t-50 107t-104 115.5q10 -104 -0.5 -189t-37 -140.5t-65 -93t-84 -52t-93.5 -11t-95 24.5q-80 36 -131.5 114t-53.5 171q-2 23 0 49.5 t4.5 52.5t13.5 56t27.5 60t46 64.5t69.5 68.5q-8 -53 -5 -102.5t17.5 -90t34 -68.5t44.5 -39t49 -2q31 13 38.5 36t-4.5 55t-29 64.5t-36 75t-26 75.5q-15 85 2 161.5t53.5 128.5t85.5 92.5t93.5 61t81.5 25.5z" />
+<glyph unicode="&#xe105;" d="M600 1094q82 0 160.5 -22.5t140 -59t116.5 -82.5t94.5 -95t68 -95t42.5 -82.5t14 -57.5t-14 -57.5t-43 -82.5t-68.5 -95t-94.5 -95t-116.5 -82.5t-140 -59t-159.5 -22.5t-159.5 22.5t-140 59t-116.5 82.5t-94.5 95t-68.5 95t-43 82.5t-14 57.5t14 57.5t42.5 82.5t68 95 t94.5 95t116.5 82.5t140 59t160.5 22.5zM888 829q-15 15 -18 12t5 -22q25 -57 25 -119q0 -124 -88 -212t-212 -88t-212 88t-88 212q0 59 23 114q8 19 4.5 22t-17.5 -12q-70 -69 -160 -184q-13 -16 -15 -40.5t9 -42.5q22 -36 47 -71t70 -82t92.5 -81t113 -58.5t133.5 -24.5 t133.5 24t113 58.5t92.5 81.5t70 81.5t47 70.5q11 18 9 42.5t-14 41.5q-90 117 -163 189zM448 727l-35 -36q-15 -15 -19.5 -38.5t4.5 -41.5q37 -68 93 -116q16 -13 38.5 -11t36.5 17l35 34q14 15 12.5 33.5t-16.5 33.5q-44 44 -89 117q-11 18 -28 20t-32 -12z" />
+<glyph unicode="&#xe106;" d="M592 0h-148l31 120q-91 20 -175.5 68.5t-143.5 106.5t-103.5 119t-66.5 110t-22 76q0 21 14 57.5t42.5 82.5t68 95t94.5 95t116.5 82.5t140 59t160.5 22.5q61 0 126 -15l32 121h148zM944 770l47 181q108 -85 176.5 -192t68.5 -159q0 -26 -19.5 -71t-59.5 -102t-93 -112 t-129 -104.5t-158 -75.5l46 173q77 49 136 117t97 131q11 18 9 42.5t-14 41.5q-54 70 -107 130zM310 824q-70 -69 -160 -184q-13 -16 -15 -40.5t9 -42.5q18 -30 39 -60t57 -70.5t74 -73t90 -61t105 -41.5l41 154q-107 18 -178.5 101.5t-71.5 193.5q0 59 23 114q8 19 4.5 22 t-17.5 -12zM448 727l-35 -36q-15 -15 -19.5 -38.5t4.5 -41.5q37 -68 93 -116q16 -13 38.5 -11t36.5 17l12 11l22 86l-3 4q-44 44 -89 117q-11 18 -28 20t-32 -12z" />
+<glyph unicode="&#xe107;" d="M-90 100l642 1066q20 31 48 28.5t48 -35.5l642 -1056q21 -32 7.5 -67.5t-50.5 -35.5h-1294q-37 0 -50.5 34t7.5 66zM155 200h345v75q0 10 7.5 17.5t17.5 7.5h150q10 0 17.5 -7.5t7.5 -17.5v-75h345l-445 723zM496 700h208q20 0 32 -14.5t8 -34.5l-58 -252 q-4 -20 -21.5 -34.5t-37.5 -14.5h-54q-20 0 -37.5 14.5t-21.5 34.5l-58 252q-4 20 8 34.5t32 14.5z" />
+<glyph unicode="&#xe108;" d="M650 1200q62 0 106 -44t44 -106v-339l363 -325q15 -14 26 -38.5t11 -44.5v-41q0 -20 -12 -26.5t-29 5.5l-359 249v-263q100 -93 100 -113v-64q0 -21 -13 -29t-32 1l-205 128l-205 -128q-19 -9 -32 -1t-13 29v64q0 20 100 113v263l-359 -249q-17 -12 -29 -5.5t-12 26.5v41 q0 20 11 44.5t26 38.5l363 325v339q0 62 44 106t106 44z" />
+<glyph unicode="&#xe109;" d="M850 1200h100q21 0 35.5 -14.5t14.5 -35.5v-50h50q21 0 35.5 -14.5t14.5 -35.5v-150h-1100v150q0 21 14.5 35.5t35.5 14.5h50v50q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-50h500v50q0 21 14.5 35.5t35.5 14.5zM1100 800v-750q0 -21 -14.5 -35.5 t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5v750h1100zM100 600v-100h100v100h-100zM300 600v-100h100v100h-100zM500 600v-100h100v100h-100zM700 600v-100h100v100h-100zM900 600v-100h100v100h-100zM100 400v-100h100v100h-100zM300 400v-100h100v100h-100zM500 400 v-100h100v100h-100zM700 400v-100h100v100h-100zM900 400v-100h100v100h-100zM100 200v-100h100v100h-100zM300 200v-100h100v100h-100zM500 200v-100h100v100h-100zM700 200v-100h100v100h-100zM900 200v-100h100v100h-100z" />
+<glyph unicode="&#xe110;" d="M1135 1165l249 -230q15 -14 15 -35t-15 -35l-249 -230q-14 -14 -24.5 -10t-10.5 25v150h-159l-600 -600h-291q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h209l600 600h241v150q0 21 10.5 25t24.5 -10zM522 819l-141 -141l-122 122h-209q-21 0 -35.5 14.5 t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h291zM1135 565l249 -230q15 -14 15 -35t-15 -35l-249 -230q-14 -14 -24.5 -10t-10.5 25v150h-241l-181 181l141 141l122 -122h159v150q0 21 10.5 25t24.5 -10z" />
+<glyph unicode="&#xe111;" d="M100 1100h1000q41 0 70.5 -29.5t29.5 -70.5v-600q0 -41 -29.5 -70.5t-70.5 -29.5h-596l-304 -300v300h-100q-41 0 -70.5 29.5t-29.5 70.5v600q0 41 29.5 70.5t70.5 29.5z" />
+<glyph unicode="&#xe112;" d="M150 1200h200q21 0 35.5 -14.5t14.5 -35.5v-250h-300v250q0 21 14.5 35.5t35.5 14.5zM850 1200h200q21 0 35.5 -14.5t14.5 -35.5v-250h-300v250q0 21 14.5 35.5t35.5 14.5zM1100 800v-300q0 -41 -3 -77.5t-15 -89.5t-32 -96t-58 -89t-89 -77t-129 -51t-174 -20t-174 20 t-129 51t-89 77t-58 89t-32 96t-15 89.5t-3 77.5v300h300v-250v-27v-42.5t1.5 -41t5 -38t10 -35t16.5 -30t25.5 -24.5t35 -19t46.5 -12t60 -4t60 4.5t46.5 12.5t35 19.5t25 25.5t17 30.5t10 35t5 38t2 40.5t-0.5 42v25v250h300z" />
+<glyph unicode="&#xe113;" d="M1100 411l-198 -199l-353 353l-353 -353l-197 199l551 551z" />
+<glyph unicode="&#xe114;" d="M1101 789l-550 -551l-551 551l198 199l353 -353l353 353z" />
+<glyph unicode="&#xe115;" d="M404 1000h746q21 0 35.5 -14.5t14.5 -35.5v-551h150q21 0 25 -10.5t-10 -24.5l-230 -249q-14 -15 -35 -15t-35 15l-230 249q-14 14 -10 24.5t25 10.5h150v401h-381zM135 984l230 -249q14 -14 10 -24.5t-25 -10.5h-150v-400h385l215 -200h-750q-21 0 -35.5 14.5 t-14.5 35.5v550h-150q-21 0 -25 10.5t10 24.5l230 249q14 15 35 15t35 -15z" />
+<glyph unicode="&#xe116;" d="M56 1200h94q17 0 31 -11t18 -27l38 -162h896q24 0 39 -18.5t10 -42.5l-100 -475q-5 -21 -27 -42.5t-55 -21.5h-633l48 -200h535q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-50v-50q0 -21 -14.5 -35.5t-35.5 -14.5t-35.5 14.5t-14.5 35.5v50h-300v-50 q0 -21 -14.5 -35.5t-35.5 -14.5t-35.5 14.5t-14.5 35.5v50h-31q-18 0 -32.5 10t-20.5 19l-5 10l-201 961h-54q-20 0 -35 14.5t-15 35.5t15 35.5t35 14.5z" />
+<glyph unicode="&#xe117;" d="M1200 1000v-100h-1200v100h200q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5h500zM0 800h1200v-800h-1200v800z" />
+<glyph unicode="&#xe118;" d="M200 800l-200 -400v600h200q0 41 29.5 70.5t70.5 29.5h300q42 0 71 -29.5t29 -70.5h500v-200h-1000zM1500 700l-300 -700h-1200l300 700h1200z" />
+<glyph unicode="&#xe119;" d="M635 1184l230 -249q14 -14 10 -24.5t-25 -10.5h-150v-601h150q21 0 25 -10.5t-10 -24.5l-230 -249q-14 -15 -35 -15t-35 15l-230 249q-14 14 -10 24.5t25 10.5h150v601h-150q-21 0 -25 10.5t10 24.5l230 249q14 15 35 15t35 -15z" />
+<glyph unicode="&#xe120;" d="M936 864l249 -229q14 -15 14 -35.5t-14 -35.5l-249 -229q-15 -15 -25.5 -10.5t-10.5 24.5v151h-600v-151q0 -20 -10.5 -24.5t-25.5 10.5l-249 229q-14 15 -14 35.5t14 35.5l249 229q15 15 25.5 10.5t10.5 -25.5v-149h600v149q0 21 10.5 25.5t25.5 -10.5z" />
+<glyph unicode="&#xe121;" d="M1169 400l-172 732q-5 23 -23 45.5t-38 22.5h-672q-20 0 -38 -20t-23 -41l-172 -739h1138zM1100 300h-1000q-41 0 -70.5 -29.5t-29.5 -70.5v-100q0 -41 29.5 -70.5t70.5 -29.5h1000q41 0 70.5 29.5t29.5 70.5v100q0 41 -29.5 70.5t-70.5 29.5zM800 100v100h100v-100h-100 zM1000 100v100h100v-100h-100z" />
+<glyph unicode="&#xe122;" d="M1150 1100q21 0 35.5 -14.5t14.5 -35.5v-850q0 -21 -14.5 -35.5t-35.5 -14.5t-35.5 14.5t-14.5 35.5v850q0 21 14.5 35.5t35.5 14.5zM1000 200l-675 200h-38l47 -276q3 -16 -5.5 -20t-29.5 -4h-7h-84q-20 0 -34.5 14t-18.5 35q-55 337 -55 351v250v6q0 16 1 23.5t6.5 14 t17.5 6.5h200l675 250v-850zM0 750v-250q-4 0 -11 0.5t-24 6t-30 15t-24 30t-11 48.5v50q0 26 10.5 46t25 30t29 16t25.5 7z" />
+<glyph unicode="&#xe123;" d="M553 1200h94q20 0 29 -10.5t3 -29.5l-18 -37q83 -19 144 -82.5t76 -140.5l63 -327l118 -173h17q19 0 33 -14.5t14 -35t-13 -40.5t-31 -27q-8 -4 -23 -9.5t-65 -19.5t-103 -25t-132.5 -20t-158.5 -9q-57 0 -115 5t-104 12t-88.5 15.5t-73.5 17.5t-54.5 16t-35.5 12l-11 4 q-18 8 -31 28t-13 40.5t14 35t33 14.5h17l118 173l63 327q15 77 76 140t144 83l-18 32q-6 19 3.5 32t28.5 13zM498 110q50 -6 102 -6q53 0 102 6q-12 -49 -39.5 -79.5t-62.5 -30.5t-63 30.5t-39 79.5z" />
+<glyph unicode="&#xe124;" d="M800 946l224 78l-78 -224l234 -45l-180 -155l180 -155l-234 -45l78 -224l-224 78l-45 -234l-155 180l-155 -180l-45 234l-224 -78l78 224l-234 45l180 155l-180 155l234 45l-78 224l224 -78l45 234l155 -180l155 180z" />
+<glyph unicode="&#xe125;" d="M650 1200h50q40 0 70 -40.5t30 -84.5v-150l-28 -125h328q40 0 70 -40.5t30 -84.5v-100q0 -45 -29 -74l-238 -344q-16 -24 -38 -40.5t-45 -16.5h-250q-7 0 -42 25t-66 50l-31 25h-61q-45 0 -72.5 18t-27.5 57v400q0 36 20 63l145 196l96 198q13 28 37.5 48t51.5 20z M650 1100l-100 -212l-150 -213v-375h100l136 -100h214l250 375v125h-450l50 225v175h-50zM50 800h100q21 0 35.5 -14.5t14.5 -35.5v-500q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v500q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe126;" d="M600 1100h250q23 0 45 -16.5t38 -40.5l238 -344q29 -29 29 -74v-100q0 -44 -30 -84.5t-70 -40.5h-328q28 -118 28 -125v-150q0 -44 -30 -84.5t-70 -40.5h-50q-27 0 -51.5 20t-37.5 48l-96 198l-145 196q-20 27 -20 63v400q0 39 27.5 57t72.5 18h61q124 100 139 100z M50 1000h100q21 0 35.5 -14.5t14.5 -35.5v-500q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v500q0 21 14.5 35.5t35.5 14.5zM636 1000l-136 -100h-100v-375l150 -213l100 -212h50v175l-50 225h450v125l-250 375h-214z" />
+<glyph unicode="&#xe127;" d="M356 873l363 230q31 16 53 -6l110 -112q13 -13 13.5 -32t-11.5 -34l-84 -121h302q84 0 138 -38t54 -110t-55 -111t-139 -39h-106l-131 -339q-6 -21 -19.5 -41t-28.5 -20h-342q-7 0 -90 81t-83 94v525q0 17 14 35.5t28 28.5zM400 792v-503l100 -89h293l131 339 q6 21 19.5 41t28.5 20h203q21 0 30.5 25t0.5 50t-31 25h-456h-7h-6h-5.5t-6 0.5t-5 1.5t-5 2t-4 2.5t-4 4t-2.5 4.5q-12 25 5 47l146 183l-86 83zM50 800h100q21 0 35.5 -14.5t14.5 -35.5v-500q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v500 q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe128;" d="M475 1103l366 -230q2 -1 6 -3.5t14 -10.5t18 -16.5t14.5 -20t6.5 -22.5v-525q0 -13 -86 -94t-93 -81h-342q-15 0 -28.5 20t-19.5 41l-131 339h-106q-85 0 -139.5 39t-54.5 111t54 110t138 38h302l-85 121q-11 15 -10.5 34t13.5 32l110 112q22 22 53 6zM370 945l146 -183 q17 -22 5 -47q-2 -2 -3.5 -4.5t-4 -4t-4 -2.5t-5 -2t-5 -1.5t-6 -0.5h-6h-6.5h-6h-475v-100h221q15 0 29 -20t20 -41l130 -339h294l106 89v503l-342 236zM1050 800h100q21 0 35.5 -14.5t14.5 -35.5v-500q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5 v500q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe129;" d="M550 1294q72 0 111 -55t39 -139v-106l339 -131q21 -6 41 -19.5t20 -28.5v-342q0 -7 -81 -90t-94 -83h-525q-17 0 -35.5 14t-28.5 28l-9 14l-230 363q-16 31 6 53l112 110q13 13 32 13.5t34 -11.5l121 -84v302q0 84 38 138t110 54zM600 972v203q0 21 -25 30.5t-50 0.5 t-25 -31v-456v-7v-6v-5.5t-0.5 -6t-1.5 -5t-2 -5t-2.5 -4t-4 -4t-4.5 -2.5q-25 -12 -47 5l-183 146l-83 -86l236 -339h503l89 100v293l-339 131q-21 6 -41 19.5t-20 28.5zM450 200h500q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-500 q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe130;" d="M350 1100h500q21 0 35.5 14.5t14.5 35.5v100q0 21 -14.5 35.5t-35.5 14.5h-500q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -21 14.5 -35.5t35.5 -14.5zM600 306v-106q0 -84 -39 -139t-111 -55t-110 54t-38 138v302l-121 -84q-15 -12 -34 -11.5t-32 13.5l-112 110 q-22 22 -6 53l230 363q1 2 3.5 6t10.5 13.5t16.5 17t20 13.5t22.5 6h525q13 0 94 -83t81 -90v-342q0 -15 -20 -28.5t-41 -19.5zM308 900l-236 -339l83 -86l183 146q22 17 47 5q2 -1 4.5 -2.5t4 -4t2.5 -4t2 -5t1.5 -5t0.5 -6v-5.5v-6v-7v-456q0 -22 25 -31t50 0.5t25 30.5 v203q0 15 20 28.5t41 19.5l339 131v293l-89 100h-503z" />
+<glyph unicode="&#xe131;" d="M600 1178q118 0 225 -45.5t184.5 -123t123 -184.5t45.5 -225t-45.5 -225t-123 -184.5t-184.5 -123t-225 -45.5t-225 45.5t-184.5 123t-123 184.5t-45.5 225t45.5 225t123 184.5t184.5 123t225 45.5zM914 632l-275 223q-16 13 -27.5 8t-11.5 -26v-137h-275 q-10 0 -17.5 -7.5t-7.5 -17.5v-150q0 -10 7.5 -17.5t17.5 -7.5h275v-137q0 -21 11.5 -26t27.5 8l275 223q16 13 16 32t-16 32z" />
+<glyph unicode="&#xe132;" d="M600 1178q118 0 225 -45.5t184.5 -123t123 -184.5t45.5 -225t-45.5 -225t-123 -184.5t-184.5 -123t-225 -45.5t-225 45.5t-184.5 123t-123 184.5t-45.5 225t45.5 225t123 184.5t184.5 123t225 45.5zM561 855l-275 -223q-16 -13 -16 -32t16 -32l275 -223q16 -13 27.5 -8 t11.5 26v137h275q10 0 17.5 7.5t7.5 17.5v150q0 10 -7.5 17.5t-17.5 7.5h-275v137q0 21 -11.5 26t-27.5 -8z" />
+<glyph unicode="&#xe133;" d="M600 1178q118 0 225 -45.5t184.5 -123t123 -184.5t45.5 -225t-45.5 -225t-123 -184.5t-184.5 -123t-225 -45.5t-225 45.5t-184.5 123t-123 184.5t-45.5 225t45.5 225t123 184.5t184.5 123t225 45.5zM855 639l-223 275q-13 16 -32 16t-32 -16l-223 -275q-13 -16 -8 -27.5 t26 -11.5h137v-275q0 -10 7.5 -17.5t17.5 -7.5h150q10 0 17.5 7.5t7.5 17.5v275h137q21 0 26 11.5t-8 27.5z" />
+<glyph unicode="&#xe134;" d="M600 1178q118 0 225 -45.5t184.5 -123t123 -184.5t45.5 -225t-45.5 -225t-123 -184.5t-184.5 -123t-225 -45.5t-225 45.5t-184.5 123t-123 184.5t-45.5 225t45.5 225t123 184.5t184.5 123t225 45.5zM675 900h-150q-10 0 -17.5 -7.5t-7.5 -17.5v-275h-137q-21 0 -26 -11.5 t8 -27.5l223 -275q13 -16 32 -16t32 16l223 275q13 16 8 27.5t-26 11.5h-137v275q0 10 -7.5 17.5t-17.5 7.5z" />
+<glyph unicode="&#xe135;" d="M600 1176q116 0 222.5 -46t184 -123.5t123.5 -184t46 -222.5t-46 -222.5t-123.5 -184t-184 -123.5t-222.5 -46t-222.5 46t-184 123.5t-123.5 184t-46 222.5t46 222.5t123.5 184t184 123.5t222.5 46zM627 1101q-15 -12 -36.5 -20.5t-35.5 -12t-43 -8t-39 -6.5 q-15 -3 -45.5 0t-45.5 -2q-20 -7 -51.5 -26.5t-34.5 -34.5q-3 -11 6.5 -22.5t8.5 -18.5q-3 -34 -27.5 -91t-29.5 -79q-9 -34 5 -93t8 -87q0 -9 17 -44.5t16 -59.5q12 0 23 -5t23.5 -15t19.5 -14q16 -8 33 -15t40.5 -15t34.5 -12q21 -9 52.5 -32t60 -38t57.5 -11 q7 -15 -3 -34t-22.5 -40t-9.5 -38q13 -21 23 -34.5t27.5 -27.5t36.5 -18q0 -7 -3.5 -16t-3.5 -14t5 -17q104 -2 221 112q30 29 46.5 47t34.5 49t21 63q-13 8 -37 8.5t-36 7.5q-15 7 -49.5 15t-51.5 19q-18 0 -41 -0.5t-43 -1.5t-42 -6.5t-38 -16.5q-51 -35 -66 -12 q-4 1 -3.5 25.5t0.5 25.5q-6 13 -26.5 17.5t-24.5 6.5q1 15 -0.5 30.5t-7 28t-18.5 11.5t-31 -21q-23 -25 -42 4q-19 28 -8 58q6 16 22 22q6 -1 26 -1.5t33.5 -4t19.5 -13.5q7 -12 18 -24t21.5 -20.5t20 -15t15.5 -10.5l5 -3q2 12 7.5 30.5t8 34.5t-0.5 32q-3 18 3.5 29 t18 22.5t15.5 24.5q6 14 10.5 35t8 31t15.5 22.5t34 22.5q-6 18 10 36q8 0 24 -1.5t24.5 -1.5t20 4.5t20.5 15.5q-10 23 -31 42.5t-37.5 29.5t-49 27t-43.5 23q0 1 2 8t3 11.5t1.5 10.5t-1 9.5t-4.5 4.5q31 -13 58.5 -14.5t38.5 2.5l12 5q5 28 -9.5 46t-36.5 24t-50 15 t-41 20q-18 -4 -37 0zM613 994q0 -17 8 -42t17 -45t9 -23q-8 1 -39.5 5.5t-52.5 10t-37 16.5q3 11 16 29.5t16 25.5q10 -10 19 -10t14 6t13.5 14.5t16.5 12.5z" />
+<glyph unicode="&#xe136;" d="M756 1157q164 92 306 -9l-259 -138l145 -232l251 126q6 -89 -34 -156.5t-117 -110.5q-60 -34 -127 -39.5t-126 16.5l-596 -596q-15 -16 -36.5 -16t-36.5 16l-111 110q-15 15 -15 36.5t15 37.5l600 599q-34 101 5.5 201.5t135.5 154.5z" />
+<glyph unicode="&#xe137;" horiz-adv-x="1220" d="M100 1196h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM1100 1096h-200v-100h200v100zM100 796h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000 q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM1100 696h-500v-100h500v100zM100 396h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM1100 296h-300v-100h300v100z " />
+<glyph unicode="&#xe138;" d="M150 1200h900q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-900q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM700 500v-300l-200 -200v500l-350 500h900z" />
+<glyph unicode="&#xe139;" d="M500 1200h200q41 0 70.5 -29.5t29.5 -70.5v-100h300q41 0 70.5 -29.5t29.5 -70.5v-400h-500v100h-200v-100h-500v400q0 41 29.5 70.5t70.5 29.5h300v100q0 41 29.5 70.5t70.5 29.5zM500 1100v-100h200v100h-200zM1200 400v-200q0 -41 -29.5 -70.5t-70.5 -29.5h-1000 q-41 0 -70.5 29.5t-29.5 70.5v200h1200z" />
+<glyph unicode="&#xe140;" d="M50 1200h300q21 0 25 -10.5t-10 -24.5l-94 -94l199 -199q7 -8 7 -18t-7 -18l-106 -106q-8 -7 -18 -7t-18 7l-199 199l-94 -94q-14 -14 -24.5 -10t-10.5 25v300q0 21 14.5 35.5t35.5 14.5zM850 1200h300q21 0 35.5 -14.5t14.5 -35.5v-300q0 -21 -10.5 -25t-24.5 10l-94 94 l-199 -199q-8 -7 -18 -7t-18 7l-106 106q-7 8 -7 18t7 18l199 199l-94 94q-14 14 -10 24.5t25 10.5zM364 470l106 -106q7 -8 7 -18t-7 -18l-199 -199l94 -94q14 -14 10 -24.5t-25 -10.5h-300q-21 0 -35.5 14.5t-14.5 35.5v300q0 21 10.5 25t24.5 -10l94 -94l199 199 q8 7 18 7t18 -7zM1071 271l94 94q14 14 24.5 10t10.5 -25v-300q0 -21 -14.5 -35.5t-35.5 -14.5h-300q-21 0 -25 10.5t10 24.5l94 94l-199 199q-7 8 -7 18t7 18l106 106q8 7 18 7t18 -7z" />
+<glyph unicode="&#xe141;" d="M596 1192q121 0 231.5 -47.5t190 -127t127 -190t47.5 -231.5t-47.5 -231.5t-127 -190.5t-190 -127t-231.5 -47t-231.5 47t-190.5 127t-127 190.5t-47 231.5t47 231.5t127 190t190.5 127t231.5 47.5zM596 1010q-112 0 -207.5 -55.5t-151 -151t-55.5 -207.5t55.5 -207.5 t151 -151t207.5 -55.5t207.5 55.5t151 151t55.5 207.5t-55.5 207.5t-151 151t-207.5 55.5zM454.5 905q22.5 0 38.5 -16t16 -38.5t-16 -39t-38.5 -16.5t-38.5 16.5t-16 39t16 38.5t38.5 16zM754.5 905q22.5 0 38.5 -16t16 -38.5t-16 -39t-38 -16.5q-14 0 -29 10l-55 -145 q17 -23 17 -51q0 -36 -25.5 -61.5t-61.5 -25.5t-61.5 25.5t-25.5 61.5q0 32 20.5 56.5t51.5 29.5l122 126l1 1q-9 14 -9 28q0 23 16 39t38.5 16zM345.5 709q22.5 0 38.5 -16t16 -38.5t-16 -38.5t-38.5 -16t-38.5 16t-16 38.5t16 38.5t38.5 16zM854.5 709q22.5 0 38.5 -16 t16 -38.5t-16 -38.5t-38.5 -16t-38.5 16t-16 38.5t16 38.5t38.5 16z" />
+<glyph unicode="&#xe142;" d="M546 173l469 470q91 91 99 192q7 98 -52 175.5t-154 94.5q-22 4 -47 4q-34 0 -66.5 -10t-56.5 -23t-55.5 -38t-48 -41.5t-48.5 -47.5q-376 -375 -391 -390q-30 -27 -45 -41.5t-37.5 -41t-32 -46.5t-16 -47.5t-1.5 -56.5q9 -62 53.5 -95t99.5 -33q74 0 125 51l548 548 q36 36 20 75q-7 16 -21.5 26t-32.5 10q-26 0 -50 -23q-13 -12 -39 -38l-341 -338q-15 -15 -35.5 -15.5t-34.5 13.5t-14 34.5t14 34.5q327 333 361 367q35 35 67.5 51.5t78.5 16.5q14 0 29 -1q44 -8 74.5 -35.5t43.5 -68.5q14 -47 2 -96.5t-47 -84.5q-12 -11 -32 -32 t-79.5 -81t-114.5 -115t-124.5 -123.5t-123 -119.5t-96.5 -89t-57 -45q-56 -27 -120 -27q-70 0 -129 32t-93 89q-48 78 -35 173t81 163l511 511q71 72 111 96q91 55 198 55q80 0 152 -33q78 -36 129.5 -103t66.5 -154q17 -93 -11 -183.5t-94 -156.5l-482 -476 q-15 -15 -36 -16t-37 14t-17.5 34t14.5 35z" />
+<glyph unicode="&#xe143;" d="M649 949q48 68 109.5 104t121.5 38.5t118.5 -20t102.5 -64t71 -100.5t27 -123q0 -57 -33.5 -117.5t-94 -124.5t-126.5 -127.5t-150 -152.5t-146 -174q-62 85 -145.5 174t-150 152.5t-126.5 127.5t-93.5 124.5t-33.5 117.5q0 64 28 123t73 100.5t104 64t119 20 t120.5 -38.5t104.5 -104zM896 972q-33 0 -64.5 -19t-56.5 -46t-47.5 -53.5t-43.5 -45.5t-37.5 -19t-36 19t-40 45.5t-43 53.5t-54 46t-65.5 19q-67 0 -122.5 -55.5t-55.5 -132.5q0 -23 13.5 -51t46 -65t57.5 -63t76 -75l22 -22q15 -14 44 -44t50.5 -51t46 -44t41 -35t23 -12 t23.5 12t42.5 36t46 44t52.5 52t44 43q4 4 12 13q43 41 63.5 62t52 55t46 55t26 46t11.5 44q0 79 -53 133.5t-120 54.5z" />
+<glyph unicode="&#xe144;" d="M776.5 1214q93.5 0 159.5 -66l141 -141q66 -66 66 -160q0 -42 -28 -95.5t-62 -87.5l-29 -29q-31 53 -77 99l-18 18l95 95l-247 248l-389 -389l212 -212l-105 -106l-19 18l-141 141q-66 66 -66 159t66 159l283 283q65 66 158.5 66zM600 706l105 105q10 -8 19 -17l141 -141 q66 -66 66 -159t-66 -159l-283 -283q-66 -66 -159 -66t-159 66l-141 141q-66 66 -66 159.5t66 159.5l55 55q29 -55 75 -102l18 -17l-95 -95l247 -248l389 389z" />
+<glyph unicode="&#xe145;" d="M603 1200q85 0 162 -15t127 -38t79 -48t29 -46v-953q0 -41 -29.5 -70.5t-70.5 -29.5h-600q-41 0 -70.5 29.5t-29.5 70.5v953q0 21 30 46.5t81 48t129 37.5t163 15zM300 1000v-700h600v700h-600zM600 254q-43 0 -73.5 -30.5t-30.5 -73.5t30.5 -73.5t73.5 -30.5t73.5 30.5 t30.5 73.5t-30.5 73.5t-73.5 30.5z" />
+<glyph unicode="&#xe146;" d="M902 1185l283 -282q15 -15 15 -36t-14.5 -35.5t-35.5 -14.5t-35 15l-36 35l-279 -267v-300l-212 210l-308 -307l-280 -203l203 280l307 308l-210 212h300l267 279l-35 36q-15 14 -15 35t14.5 35.5t35.5 14.5t35 -15z" />
+<glyph unicode="&#xe148;" d="M700 1248v-78q38 -5 72.5 -14.5t75.5 -31.5t71 -53.5t52 -84t24 -118.5h-159q-4 36 -10.5 59t-21 45t-40 35.5t-64.5 20.5v-307l64 -13q34 -7 64 -16.5t70 -32t67.5 -52.5t47.5 -80t20 -112q0 -139 -89 -224t-244 -97v-77h-100v79q-150 16 -237 103q-40 40 -52.5 93.5 t-15.5 139.5h139q5 -77 48.5 -126t117.5 -65v335l-27 8q-46 14 -79 26.5t-72 36t-63 52t-40 72.5t-16 98q0 70 25 126t67.5 92t94.5 57t110 27v77h100zM600 754v274q-29 -4 -50 -11t-42 -21.5t-31.5 -41.5t-10.5 -65q0 -29 7 -50.5t16.5 -34t28.5 -22.5t31.5 -14t37.5 -10 q9 -3 13 -4zM700 547v-310q22 2 42.5 6.5t45 15.5t41.5 27t29 42t12 59.5t-12.5 59.5t-38 44.5t-53 31t-66.5 24.5z" />
+<glyph unicode="&#xe149;" d="M561 1197q84 0 160.5 -40t123.5 -109.5t47 -147.5h-153q0 40 -19.5 71.5t-49.5 48.5t-59.5 26t-55.5 9q-37 0 -79 -14.5t-62 -35.5q-41 -44 -41 -101q0 -26 13.5 -63t26.5 -61t37 -66q6 -9 9 -14h241v-100h-197q8 -50 -2.5 -115t-31.5 -95q-45 -62 -99 -112 q34 10 83 17.5t71 7.5q32 1 102 -16t104 -17q83 0 136 30l50 -147q-31 -19 -58 -30.5t-55 -15.5t-42 -4.5t-46 -0.5q-23 0 -76 17t-111 32.5t-96 11.5q-39 -3 -82 -16t-67 -25l-23 -11l-55 145q4 3 16 11t15.5 10.5t13 9t15.5 12t14.5 14t17.5 18.5q48 55 54 126.5 t-30 142.5h-221v100h166q-23 47 -44 104q-7 20 -12 41.5t-6 55.5t6 66.5t29.5 70.5t58.5 71q97 88 263 88z" />
+<glyph unicode="&#xe150;" d="M400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM935 1184l230 -249q14 -14 10 -24.5t-25 -10.5h-150v-900h-200v900h-150q-21 0 -25 10.5t10 24.5l230 249q14 15 35 15t35 -15z" />
+<glyph unicode="&#xe151;" d="M1000 700h-100v100h-100v-100h-100v500h300v-500zM400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM801 1100v-200h100v200h-100zM1000 350l-200 -250h200v-100h-300v150l200 250h-200v100h300v-150z " />
+<glyph unicode="&#xe152;" d="M400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM1000 1050l-200 -250h200v-100h-300v150l200 250h-200v100h300v-150zM1000 0h-100v100h-100v-100h-100v500h300v-500zM801 400v-200h100v200h-100z " />
+<glyph unicode="&#xe153;" d="M400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM1000 700h-100v400h-100v100h200v-500zM1100 0h-100v100h-200v400h300v-500zM901 400v-200h100v200h-100z" />
+<glyph unicode="&#xe154;" d="M400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM1100 700h-100v100h-200v400h300v-500zM901 1100v-200h100v200h-100zM1000 0h-100v400h-100v100h200v-500z" />
+<glyph unicode="&#xe155;" d="M400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM900 1000h-200v200h200v-200zM1000 700h-300v200h300v-200zM1100 400h-400v200h400v-200zM1200 100h-500v200h500v-200z" />
+<glyph unicode="&#xe156;" d="M400 300h150q21 0 25 -11t-10 -25l-230 -250q-14 -15 -35 -15t-35 15l-230 250q-14 14 -10 25t25 11h150v900h200v-900zM1200 1000h-500v200h500v-200zM1100 700h-400v200h400v-200zM1000 400h-300v200h300v-200zM900 100h-200v200h200v-200z" />
+<glyph unicode="&#xe157;" d="M350 1100h400q162 0 256 -93.5t94 -256.5v-400q0 -165 -93.5 -257.5t-256.5 -92.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400q0 165 92.5 257.5t257.5 92.5zM800 900h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5 v500q0 41 -29.5 70.5t-70.5 29.5z" />
+<glyph unicode="&#xe158;" d="M350 1100h400q165 0 257.5 -92.5t92.5 -257.5v-400q0 -165 -92.5 -257.5t-257.5 -92.5h-400q-163 0 -256.5 92.5t-93.5 257.5v400q0 163 94 256.5t256 93.5zM800 900h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5 v500q0 41 -29.5 70.5t-70.5 29.5zM440 770l253 -190q17 -12 17 -30t-17 -30l-253 -190q-16 -12 -28 -6.5t-12 26.5v400q0 21 12 26.5t28 -6.5z" />
+<glyph unicode="&#xe159;" d="M350 1100h400q163 0 256.5 -94t93.5 -256v-400q0 -165 -92.5 -257.5t-257.5 -92.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400q0 163 92.5 256.5t257.5 93.5zM800 900h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5 v500q0 41 -29.5 70.5t-70.5 29.5zM350 700h400q21 0 26.5 -12t-6.5 -28l-190 -253q-12 -17 -30 -17t-30 17l-190 253q-12 16 -6.5 28t26.5 12z" />
+<glyph unicode="&#xe160;" d="M350 1100h400q165 0 257.5 -92.5t92.5 -257.5v-400q0 -163 -92.5 -256.5t-257.5 -93.5h-400q-163 0 -256.5 94t-93.5 256v400q0 165 92.5 257.5t257.5 92.5zM800 900h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5 v500q0 41 -29.5 70.5t-70.5 29.5zM580 693l190 -253q12 -16 6.5 -28t-26.5 -12h-400q-21 0 -26.5 12t6.5 28l190 253q12 17 30 17t30 -17z" />
+<glyph unicode="&#xe161;" d="M550 1100h400q165 0 257.5 -92.5t92.5 -257.5v-400q0 -165 -92.5 -257.5t-257.5 -92.5h-400q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h450q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5h-450q-21 0 -35.5 14.5t-14.5 35.5v100 q0 21 14.5 35.5t35.5 14.5zM338 867l324 -284q16 -14 16 -33t-16 -33l-324 -284q-16 -14 -27 -9t-11 26v150h-250q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h250v150q0 21 11 26t27 -9z" />
+<glyph unicode="&#xe162;" d="M793 1182l9 -9q8 -10 5 -27q-3 -11 -79 -225.5t-78 -221.5l300 1q24 0 32.5 -17.5t-5.5 -35.5q-1 0 -133.5 -155t-267 -312.5t-138.5 -162.5q-12 -15 -26 -15h-9l-9 8q-9 11 -4 32q2 9 42 123.5t79 224.5l39 110h-302q-23 0 -31 19q-10 21 6 41q75 86 209.5 237.5 t228 257t98.5 111.5q9 16 25 16h9z" />
+<glyph unicode="&#xe163;" d="M350 1100h400q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-450q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h450q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400 q0 165 92.5 257.5t257.5 92.5zM938 867l324 -284q16 -14 16 -33t-16 -33l-324 -284q-16 -14 -27 -9t-11 26v150h-250q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h250v150q0 21 11 26t27 -9z" />
+<glyph unicode="&#xe164;" d="M750 1200h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -10.5 -25t-24.5 10l-109 109l-312 -312q-15 -15 -35.5 -15t-35.5 15l-141 141q-15 15 -15 35.5t15 35.5l312 312l-109 109q-14 14 -10 24.5t25 10.5zM456 900h-156q-41 0 -70.5 -29.5t-29.5 -70.5v-500 q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v148l200 200v-298q0 -165 -93.5 -257.5t-256.5 -92.5h-400q-165 0 -257.5 92.5t-92.5 257.5v400q0 165 92.5 257.5t257.5 92.5h300z" />
+<glyph unicode="&#xe165;" d="M600 1186q119 0 227.5 -46.5t187 -125t125 -187t46.5 -227.5t-46.5 -227.5t-125 -187t-187 -125t-227.5 -46.5t-227.5 46.5t-187 125t-125 187t-46.5 227.5t46.5 227.5t125 187t187 125t227.5 46.5zM600 1022q-115 0 -212 -56.5t-153.5 -153.5t-56.5 -212t56.5 -212 t153.5 -153.5t212 -56.5t212 56.5t153.5 153.5t56.5 212t-56.5 212t-153.5 153.5t-212 56.5zM600 794q80 0 137 -57t57 -137t-57 -137t-137 -57t-137 57t-57 137t57 137t137 57z" />
+<glyph unicode="&#xe166;" d="M450 1200h200q21 0 35.5 -14.5t14.5 -35.5v-350h245q20 0 25 -11t-9 -26l-383 -426q-14 -15 -33.5 -15t-32.5 15l-379 426q-13 15 -8.5 26t25.5 11h250v350q0 21 14.5 35.5t35.5 14.5zM50 300h1000q21 0 35.5 -14.5t14.5 -35.5v-250h-1100v250q0 21 14.5 35.5t35.5 14.5z M900 200v-50h100v50h-100z" />
+<glyph unicode="&#xe167;" d="M583 1182l378 -435q14 -15 9 -31t-26 -16h-244v-250q0 -20 -17 -35t-39 -15h-200q-20 0 -32 14.5t-12 35.5v250h-250q-20 0 -25.5 16.5t8.5 31.5l383 431q14 16 33.5 17t33.5 -14zM50 300h1000q21 0 35.5 -14.5t14.5 -35.5v-250h-1100v250q0 21 14.5 35.5t35.5 14.5z M900 200v-50h100v50h-100z" />
+<glyph unicode="&#xe168;" d="M396 723l369 369q7 7 17.5 7t17.5 -7l139 -139q7 -8 7 -18.5t-7 -17.5l-525 -525q-7 -8 -17.5 -8t-17.5 8l-292 291q-7 8 -7 18t7 18l139 139q8 7 18.5 7t17.5 -7zM50 300h1000q21 0 35.5 -14.5t14.5 -35.5v-250h-1100v250q0 21 14.5 35.5t35.5 14.5zM900 200v-50h100v50 h-100z" />
+<glyph unicode="&#xe169;" d="M135 1023l142 142q14 14 35 14t35 -14l77 -77l-212 -212l-77 76q-14 15 -14 36t14 35zM655 855l210 210q14 14 24.5 10t10.5 -25l-2 -599q-1 -20 -15.5 -35t-35.5 -15l-597 -1q-21 0 -25 10.5t10 24.5l208 208l-154 155l212 212zM50 300h1000q21 0 35.5 -14.5t14.5 -35.5 v-250h-1100v250q0 21 14.5 35.5t35.5 14.5zM900 200v-50h100v50h-100z" />
+<glyph unicode="&#xe170;" d="M350 1200l599 -2q20 -1 35 -15.5t15 -35.5l1 -597q0 -21 -10.5 -25t-24.5 10l-208 208l-155 -154l-212 212l155 154l-210 210q-14 14 -10 24.5t25 10.5zM524 512l-76 -77q-15 -14 -36 -14t-35 14l-142 142q-14 14 -14 35t14 35l77 77zM50 300h1000q21 0 35.5 -14.5 t14.5 -35.5v-250h-1100v250q0 21 14.5 35.5t35.5 14.5zM900 200v-50h100v50h-100z" />
+<glyph unicode="&#xe171;" d="M1200 103l-483 276l-314 -399v423h-399l1196 796v-1096zM483 424v-230l683 953z" />
+<glyph unicode="&#xe172;" d="M1100 1000v-850q0 -21 -14.5 -35.5t-35.5 -14.5h-150v400h-700v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100zM700 1000h-100v200h100v-200z" />
+<glyph unicode="&#xe173;" d="M1100 1000l-2 -149l-299 -299l-95 95q-9 9 -21.5 9t-21.5 -9l-149 -147h-312v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100zM700 1000h-100v200h100v-200zM1132 638l106 -106q7 -7 7 -17.5t-7 -17.5l-420 -421q-8 -7 -18 -7 t-18 7l-202 203q-8 7 -8 17.5t8 17.5l106 106q7 8 17.5 8t17.5 -8l79 -79l297 297q7 7 17.5 7t17.5 -7z" />
+<glyph unicode="&#xe174;" d="M1100 1000v-269l-103 -103l-134 134q-15 15 -33.5 16.5t-34.5 -12.5l-266 -266h-329v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100zM700 1000h-100v200h100v-200zM1202 572l70 -70q15 -15 15 -35.5t-15 -35.5l-131 -131 l131 -131q15 -15 15 -35.5t-15 -35.5l-70 -70q-15 -15 -35.5 -15t-35.5 15l-131 131l-131 -131q-15 -15 -35.5 -15t-35.5 15l-70 70q-15 15 -15 35.5t15 35.5l131 131l-131 131q-15 15 -15 35.5t15 35.5l70 70q15 15 35.5 15t35.5 -15l131 -131l131 131q15 15 35.5 15 t35.5 -15z" />
+<glyph unicode="&#xe175;" d="M1100 1000v-300h-350q-21 0 -35.5 -14.5t-14.5 -35.5v-150h-500v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100zM700 1000h-100v200h100v-200zM850 600h100q21 0 35.5 -14.5t14.5 -35.5v-250h150q21 0 25 -10.5t-10 -24.5 l-230 -230q-14 -14 -35 -14t-35 14l-230 230q-14 14 -10 24.5t25 10.5h150v250q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe176;" d="M1100 1000v-400l-165 165q-14 15 -35 15t-35 -15l-263 -265h-402v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100zM700 1000h-100v200h100v-200zM935 565l230 -229q14 -15 10 -25.5t-25 -10.5h-150v-250q0 -20 -14.5 -35 t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35v250h-150q-21 0 -25 10.5t10 25.5l230 229q14 15 35 15t35 -15z" />
+<glyph unicode="&#xe177;" d="M50 1100h1100q21 0 35.5 -14.5t14.5 -35.5v-150h-1200v150q0 21 14.5 35.5t35.5 14.5zM1200 800v-550q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v550h1200zM100 500v-200h400v200h-400z" />
+<glyph unicode="&#xe178;" d="M935 1165l248 -230q14 -14 14 -35t-14 -35l-248 -230q-14 -14 -24.5 -10t-10.5 25v150h-400v200h400v150q0 21 10.5 25t24.5 -10zM200 800h-50q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h50v-200zM400 800h-100v200h100v-200zM18 435l247 230 q14 14 24.5 10t10.5 -25v-150h400v-200h-400v-150q0 -21 -10.5 -25t-24.5 10l-247 230q-15 14 -15 35t15 35zM900 300h-100v200h100v-200zM1000 500h51q20 0 34.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-34.5 -14.5h-51v200z" />
+<glyph unicode="&#xe179;" d="M862 1073l276 116q25 18 43.5 8t18.5 -41v-1106q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v397q-4 1 -11 5t-24 17.5t-30 29t-24 42t-11 56.5v359q0 31 18.5 65t43.5 52zM550 1200q22 0 34.5 -12.5t14.5 -24.5l1 -13v-450q0 -28 -10.5 -59.5 t-25 -56t-29 -45t-25.5 -31.5l-10 -11v-447q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v447q-4 4 -11 11.5t-24 30.5t-30 46t-24 55t-11 60v450q0 2 0.5 5.5t4 12t8.5 15t14.5 12t22.5 5.5q20 0 32.5 -12.5t14.5 -24.5l3 -13v-350h100v350v5.5t2.5 12 t7 15t15 12t25.5 5.5q23 0 35.5 -12.5t13.5 -24.5l1 -13v-350h100v350q0 2 0.5 5.5t3 12t7 15t15 12t24.5 5.5z" />
+<glyph unicode="&#xe180;" d="M1200 1100v-56q-4 0 -11 -0.5t-24 -3t-30 -7.5t-24 -15t-11 -24v-888q0 -22 25 -34.5t50 -13.5l25 -2v-56h-400v56q75 0 87.5 6.5t12.5 43.5v394h-500v-394q0 -37 12.5 -43.5t87.5 -6.5v-56h-400v56q4 0 11 0.5t24 3t30 7.5t24 15t11 24v888q0 22 -25 34.5t-50 13.5 l-25 2v56h400v-56q-75 0 -87.5 -6.5t-12.5 -43.5v-394h500v394q0 37 -12.5 43.5t-87.5 6.5v56h400z" />
+<glyph unicode="&#xe181;" d="M675 1000h375q21 0 35.5 -14.5t14.5 -35.5v-150h-105l-295 -98v98l-200 200h-400l100 100h375zM100 900h300q41 0 70.5 -29.5t29.5 -70.5v-500q0 -41 -29.5 -70.5t-70.5 -29.5h-300q-41 0 -70.5 29.5t-29.5 70.5v500q0 41 29.5 70.5t70.5 29.5zM100 800v-200h300v200 h-300zM1100 535l-400 -133v163l400 133v-163zM100 500v-200h300v200h-300zM1100 398v-248q0 -21 -14.5 -35.5t-35.5 -14.5h-375l-100 -100h-375l-100 100h400l200 200h105z" />
+<glyph unicode="&#xe182;" d="M17 1007l162 162q17 17 40 14t37 -22l139 -194q14 -20 11 -44.5t-20 -41.5l-119 -118q102 -142 228 -268t267 -227l119 118q17 17 42.5 19t44.5 -12l192 -136q19 -14 22.5 -37.5t-13.5 -40.5l-163 -162q-3 -1 -9.5 -1t-29.5 2t-47.5 6t-62.5 14.5t-77.5 26.5t-90 42.5 t-101.5 60t-111 83t-119 108.5q-74 74 -133.5 150.5t-94.5 138.5t-60 119.5t-34.5 100t-15 74.5t-4.5 48z" />
+<glyph unicode="&#xe183;" d="M600 1100q92 0 175 -10.5t141.5 -27t108.5 -36.5t81.5 -40t53.5 -37t31 -27l9 -10v-200q0 -21 -14.5 -33t-34.5 -9l-202 34q-20 3 -34.5 20t-14.5 38v146q-141 24 -300 24t-300 -24v-146q0 -21 -14.5 -38t-34.5 -20l-202 -34q-20 -3 -34.5 9t-14.5 33v200q3 4 9.5 10.5 t31 26t54 37.5t80.5 39.5t109 37.5t141 26.5t175 10.5zM600 795q56 0 97 -9.5t60 -23.5t30 -28t12 -24l1 -10v-50l365 -303q14 -15 24.5 -40t10.5 -45v-212q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v212q0 20 10.5 45t24.5 40l365 303v50 q0 4 1 10.5t12 23t30 29t60 22.5t97 10z" />
+<glyph unicode="&#xe184;" d="M1100 700l-200 -200h-600l-200 200v500h200v-200h200v200h200v-200h200v200h200v-500zM250 400h700q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-12l137 -100h-950l137 100h-12q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM50 100h1100q21 0 35.5 -14.5 t14.5 -35.5v-50h-1200v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe185;" d="M700 1100h-100q-41 0 -70.5 -29.5t-29.5 -70.5v-1000h300v1000q0 41 -29.5 70.5t-70.5 29.5zM1100 800h-100q-41 0 -70.5 -29.5t-29.5 -70.5v-700h300v700q0 41 -29.5 70.5t-70.5 29.5zM400 0h-300v400q0 41 29.5 70.5t70.5 29.5h100q41 0 70.5 -29.5t29.5 -70.5v-400z " />
+<glyph unicode="&#xe186;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM500 700h-200v-100h200v-300h-300v100h200v100h-200v300h300v-100zM900 700v-300l-100 -100h-200v500h200z M700 700v-300h100v300h-100z" />
+<glyph unicode="&#xe187;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM500 300h-100v200h-100v-200h-100v500h100v-200h100v200h100v-500zM900 700v-300l-100 -100h-200v500h200z M700 700v-300h100v300h-100z" />
+<glyph unicode="&#xe188;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM500 700h-200v-300h200v-100h-300v500h300v-100zM900 700h-200v-300h200v-100h-300v500h300v-100z" />
+<glyph unicode="&#xe189;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM500 400l-300 150l300 150v-300zM900 550l-300 -150v300z" />
+<glyph unicode="&#xe190;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM900 300h-700v500h700v-500zM800 700h-130q-38 0 -66.5 -43t-28.5 -108t27 -107t68 -42h130v300zM300 700v-300 h130q41 0 68 42t27 107t-28.5 108t-66.5 43h-130z" />
+<glyph unicode="&#xe191;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM500 700h-200v-100h200v-300h-300v100h200v100h-200v300h300v-100zM900 300h-100v400h-100v100h200v-500z M700 300h-100v100h100v-100z" />
+<glyph unicode="&#xe192;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM300 700h200v-400h-300v500h100v-100zM900 300h-100v400h-100v100h200v-500zM300 600v-200h100v200h-100z M700 300h-100v100h100v-100z" />
+<glyph unicode="&#xe193;" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM100 900v-700h900v700h-900zM500 500l-199 -200h-100v50l199 200v150h-200v100h300v-300zM900 300h-100v400h-100v100h200v-500zM701 300h-100 v100h100v-100z" />
+<glyph unicode="&#xe194;" d="M600 1191q120 0 229.5 -47t188.5 -126t126 -188.5t47 -229.5t-47 -229.5t-126 -188.5t-188.5 -126t-229.5 -47t-229.5 47t-188.5 126t-126 188.5t-47 229.5t47 229.5t126 188.5t188.5 126t229.5 47zM600 1021q-114 0 -211 -56.5t-153.5 -153.5t-56.5 -211t56.5 -211 t153.5 -153.5t211 -56.5t211 56.5t153.5 153.5t56.5 211t-56.5 211t-153.5 153.5t-211 56.5zM800 700h-300v-200h300v-100h-300l-100 100v200l100 100h300v-100z" />
+<glyph unicode="&#xe195;" d="M600 1191q120 0 229.5 -47t188.5 -126t126 -188.5t47 -229.5t-47 -229.5t-126 -188.5t-188.5 -126t-229.5 -47t-229.5 47t-188.5 126t-126 188.5t-47 229.5t47 229.5t126 188.5t188.5 126t229.5 47zM600 1021q-114 0 -211 -56.5t-153.5 -153.5t-56.5 -211t56.5 -211 t153.5 -153.5t211 -56.5t211 56.5t153.5 153.5t56.5 211t-56.5 211t-153.5 153.5t-211 56.5zM800 700v-100l-50 -50l100 -100v-50h-100l-100 100h-150v-100h-100v400h300zM500 700v-100h200v100h-200z" />
+<glyph unicode="&#xe197;" d="M503 1089q110 0 200.5 -59.5t134.5 -156.5q44 14 90 14q120 0 205 -86.5t85 -207t-85 -207t-205 -86.5h-128v250q0 21 -14.5 35.5t-35.5 14.5h-300q-21 0 -35.5 -14.5t-14.5 -35.5v-250h-222q-80 0 -136 57.5t-56 136.5q0 69 43 122.5t108 67.5q-2 19 -2 37q0 100 49 185 t134 134t185 49zM525 500h150q10 0 17.5 -7.5t7.5 -17.5v-275h137q21 0 26 -11.5t-8 -27.5l-223 -244q-13 -16 -32 -16t-32 16l-223 244q-13 16 -8 27.5t26 11.5h137v275q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe198;" d="M502 1089q110 0 201 -59.5t135 -156.5q43 15 89 15q121 0 206 -86.5t86 -206.5q0 -99 -60 -181t-150 -110l-378 360q-13 16 -31.5 16t-31.5 -16l-381 -365h-9q-79 0 -135.5 57.5t-56.5 136.5q0 69 43 122.5t108 67.5q-2 19 -2 38q0 100 49 184.5t133.5 134t184.5 49.5z M632 467l223 -228q13 -16 8 -27.5t-26 -11.5h-137v-275q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v275h-137q-21 0 -26 11.5t8 27.5q199 204 223 228q19 19 31.5 19t32.5 -19z" />
+<glyph unicode="&#xe199;" d="M700 100v100h400l-270 300h170l-270 300h170l-300 333l-300 -333h170l-270 -300h170l-270 -300h400v-100h-50q-21 0 -35.5 -14.5t-14.5 -35.5v-50h400v50q0 21 -14.5 35.5t-35.5 14.5h-50z" />
+<glyph unicode="&#xe200;" d="M600 1179q94 0 167.5 -56.5t99.5 -145.5q89 -6 150.5 -71.5t61.5 -155.5q0 -61 -29.5 -112.5t-79.5 -82.5q9 -29 9 -55q0 -74 -52.5 -126.5t-126.5 -52.5q-55 0 -100 30v-251q21 0 35.5 -14.5t14.5 -35.5v-50h-300v50q0 21 14.5 35.5t35.5 14.5v251q-45 -30 -100 -30 q-74 0 -126.5 52.5t-52.5 126.5q0 18 4 38q-47 21 -75.5 65t-28.5 97q0 74 52.5 126.5t126.5 52.5q5 0 23 -2q0 2 -1 10t-1 13q0 116 81.5 197.5t197.5 81.5z" />
+<glyph unicode="&#xe201;" d="M1010 1010q111 -111 150.5 -260.5t0 -299t-150.5 -260.5q-83 -83 -191.5 -126.5t-218.5 -43.5t-218.5 43.5t-191.5 126.5q-111 111 -150.5 260.5t0 299t150.5 260.5q83 83 191.5 126.5t218.5 43.5t218.5 -43.5t191.5 -126.5zM476 1065q-4 0 -8 -1q-121 -34 -209.5 -122.5 t-122.5 -209.5q-4 -12 2.5 -23t18.5 -14l36 -9q3 -1 7 -1q23 0 29 22q27 96 98 166q70 71 166 98q11 3 17.5 13.5t3.5 22.5l-9 35q-3 13 -14 19q-7 4 -15 4zM512 920q-4 0 -9 -2q-80 -24 -138.5 -82.5t-82.5 -138.5q-4 -13 2 -24t19 -14l34 -9q4 -1 8 -1q22 0 28 21 q18 58 58.5 98.5t97.5 58.5q12 3 18 13.5t3 21.5l-9 35q-3 12 -14 19q-7 4 -15 4zM719.5 719.5q-49.5 49.5 -119.5 49.5t-119.5 -49.5t-49.5 -119.5t49.5 -119.5t119.5 -49.5t119.5 49.5t49.5 119.5t-49.5 119.5zM855 551q-22 0 -28 -21q-18 -58 -58.5 -98.5t-98.5 -57.5 q-11 -4 -17 -14.5t-3 -21.5l9 -35q3 -12 14 -19q7 -4 15 -4q4 0 9 2q80 24 138.5 82.5t82.5 138.5q4 13 -2.5 24t-18.5 14l-34 9q-4 1 -8 1zM1000 515q-23 0 -29 -22q-27 -96 -98 -166q-70 -71 -166 -98q-11 -3 -17.5 -13.5t-3.5 -22.5l9 -35q3 -13 14 -19q7 -4 15 -4 q4 0 8 1q121 34 209.5 122.5t122.5 209.5q4 12 -2.5 23t-18.5 14l-36 9q-3 1 -7 1z" />
+<glyph unicode="&#xe202;" d="M700 800h300v-380h-180v200h-340v-200h-380v755q0 10 7.5 17.5t17.5 7.5h575v-400zM1000 900h-200v200zM700 300h162l-212 -212l-212 212h162v200h100v-200zM520 0h-395q-10 0 -17.5 7.5t-7.5 17.5v395zM1000 220v-195q0 -10 -7.5 -17.5t-17.5 -7.5h-195z" />
+<glyph unicode="&#xe203;" d="M700 800h300v-520l-350 350l-550 -550v1095q0 10 7.5 17.5t17.5 7.5h575v-400zM1000 900h-200v200zM862 200h-162v-200h-100v200h-162l212 212zM480 0h-355q-10 0 -17.5 7.5t-7.5 17.5v55h380v-80zM1000 80v-55q0 -10 -7.5 -17.5t-17.5 -7.5h-155v80h180z" />
+<glyph unicode="&#xe204;" d="M1162 800h-162v-200h100l100 -100h-300v300h-162l212 212zM200 800h200q27 0 40 -2t29.5 -10.5t23.5 -30t7 -57.5h300v-100h-600l-200 -350v450h100q0 36 7 57.5t23.5 30t29.5 10.5t40 2zM800 400h240l-240 -400h-800l300 500h500v-100z" />
+<glyph unicode="&#xe205;" d="M650 1100h100q21 0 35.5 -14.5t14.5 -35.5v-50h50q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-300q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h50v50q0 21 14.5 35.5t35.5 14.5zM1000 850v150q41 0 70.5 -29.5t29.5 -70.5v-800 q0 -41 -29.5 -70.5t-70.5 -29.5h-600q-1 0 -20 4l246 246l-326 326v324q0 41 29.5 70.5t70.5 29.5v-150q0 -62 44 -106t106 -44h300q62 0 106 44t44 106zM412 250l-212 -212v162h-200v100h200v162z" />
+<glyph unicode="&#xe206;" d="M450 1100h100q21 0 35.5 -14.5t14.5 -35.5v-50h50q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-300q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h50v50q0 21 14.5 35.5t35.5 14.5zM800 850v150q41 0 70.5 -29.5t29.5 -70.5v-500 h-200v-300h200q0 -36 -7 -57.5t-23.5 -30t-29.5 -10.5t-40 -2h-600q-41 0 -70.5 29.5t-29.5 70.5v800q0 41 29.5 70.5t70.5 29.5v-150q0 -62 44 -106t106 -44h300q62 0 106 44t44 106zM1212 250l-212 -212v162h-200v100h200v162z" />
+<glyph unicode="&#xe209;" d="M658 1197l637 -1104q23 -38 7 -65.5t-60 -27.5h-1276q-44 0 -60 27.5t7 65.5l637 1104q22 39 54 39t54 -39zM704 800h-208q-20 0 -32 -14.5t-8 -34.5l58 -302q4 -20 21.5 -34.5t37.5 -14.5h54q20 0 37.5 14.5t21.5 34.5l58 302q4 20 -8 34.5t-32 14.5zM500 300v-100h200 v100h-200z" />
+<glyph unicode="&#xe210;" d="M425 1100h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5zM425 800h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5 t17.5 7.5zM825 800h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5zM25 500h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150 q0 10 7.5 17.5t17.5 7.5zM425 500h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5zM825 500h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5 v150q0 10 7.5 17.5t17.5 7.5zM25 200h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5zM425 200h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5 t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5zM825 200h250q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-250q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe211;" d="M700 1200h100v-200h-100v-100h350q62 0 86.5 -39.5t-3.5 -94.5l-66 -132q-41 -83 -81 -134h-772q-40 51 -81 134l-66 132q-28 55 -3.5 94.5t86.5 39.5h350v100h-100v200h100v100h200v-100zM250 400h700q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-12l137 -100 h-950l138 100h-13q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM50 100h1100q21 0 35.5 -14.5t14.5 -35.5v-50h-1200v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe212;" d="M600 1300q40 0 68.5 -29.5t28.5 -70.5h-194q0 41 28.5 70.5t68.5 29.5zM443 1100h314q18 -37 18 -75q0 -8 -3 -25h328q41 0 44.5 -16.5t-30.5 -38.5l-175 -145h-678l-178 145q-34 22 -29 38.5t46 16.5h328q-3 17 -3 25q0 38 18 75zM250 700h700q21 0 35.5 -14.5 t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-150v-200l275 -200h-950l275 200v200h-150q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM50 100h1100q21 0 35.5 -14.5t14.5 -35.5v-50h-1200v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe213;" d="M600 1181q75 0 128 -53t53 -128t-53 -128t-128 -53t-128 53t-53 128t53 128t128 53zM602 798h46q34 0 55.5 -28.5t21.5 -86.5q0 -76 39 -183h-324q39 107 39 183q0 58 21.5 86.5t56.5 28.5h45zM250 400h700q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-13 l138 -100h-950l137 100h-12q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM50 100h1100q21 0 35.5 -14.5t14.5 -35.5v-50h-1200v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe214;" d="M600 1300q47 0 92.5 -53.5t71 -123t25.5 -123.5q0 -78 -55.5 -133.5t-133.5 -55.5t-133.5 55.5t-55.5 133.5q0 62 34 143l144 -143l111 111l-163 163q34 26 63 26zM602 798h46q34 0 55.5 -28.5t21.5 -86.5q0 -76 39 -183h-324q39 107 39 183q0 58 21.5 86.5t56.5 28.5h45 zM250 400h700q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-13l138 -100h-950l137 100h-12q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM50 100h1100q21 0 35.5 -14.5t14.5 -35.5v-50h-1200v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe215;" d="M600 1200l300 -161v-139h-300q0 -57 18.5 -108t50 -91.5t63 -72t70 -67.5t57.5 -61h-530q-60 83 -90.5 177.5t-30.5 178.5t33 164.5t87.5 139.5t126 96.5t145.5 41.5v-98zM250 400h700q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-13l138 -100h-950l137 100 h-12q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5zM50 100h1100q21 0 35.5 -14.5t14.5 -35.5v-50h-1200v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe216;" d="M600 1300q41 0 70.5 -29.5t29.5 -70.5v-78q46 -26 73 -72t27 -100v-50h-400v50q0 54 27 100t73 72v78q0 41 29.5 70.5t70.5 29.5zM400 800h400q54 0 100 -27t72 -73h-172v-100h200v-100h-200v-100h200v-100h-200v-100h200q0 -83 -58.5 -141.5t-141.5 -58.5h-400 q-83 0 -141.5 58.5t-58.5 141.5v400q0 83 58.5 141.5t141.5 58.5z" />
+<glyph unicode="&#xe218;" d="M150 1100h900q21 0 35.5 -14.5t14.5 -35.5v-500q0 -21 -14.5 -35.5t-35.5 -14.5h-900q-21 0 -35.5 14.5t-14.5 35.5v500q0 21 14.5 35.5t35.5 14.5zM125 400h950q10 0 17.5 -7.5t7.5 -17.5v-50q0 -10 -7.5 -17.5t-17.5 -7.5h-283l224 -224q13 -13 13 -31.5t-13 -32 t-31.5 -13.5t-31.5 13l-88 88h-524l-87 -88q-13 -13 -32 -13t-32 13.5t-13 32t13 31.5l224 224h-289q-10 0 -17.5 7.5t-7.5 17.5v50q0 10 7.5 17.5t17.5 7.5zM541 300l-100 -100h324l-100 100h-124z" />
+<glyph unicode="&#xe219;" d="M200 1100h800q83 0 141.5 -58.5t58.5 -141.5v-200h-100q0 41 -29.5 70.5t-70.5 29.5h-250q-41 0 -70.5 -29.5t-29.5 -70.5h-100q0 41 -29.5 70.5t-70.5 29.5h-250q-41 0 -70.5 -29.5t-29.5 -70.5h-100v200q0 83 58.5 141.5t141.5 58.5zM100 600h1000q41 0 70.5 -29.5 t29.5 -70.5v-300h-1200v300q0 41 29.5 70.5t70.5 29.5zM300 100v-50q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v50h200zM1100 100v-50q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v50h200z" />
+<glyph unicode="&#xe221;" d="M480 1165l682 -683q31 -31 31 -75.5t-31 -75.5l-131 -131h-481l-517 518q-32 31 -32 75.5t32 75.5l295 296q31 31 75.5 31t76.5 -31zM108 794l342 -342l303 304l-341 341zM250 100h800q21 0 35.5 -14.5t14.5 -35.5v-50h-900v50q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe223;" d="M1057 647l-189 506q-8 19 -27.5 33t-40.5 14h-400q-21 0 -40.5 -14t-27.5 -33l-189 -506q-8 -19 1.5 -33t30.5 -14h625v-150q0 -21 14.5 -35.5t35.5 -14.5t35.5 14.5t14.5 35.5v150h125q21 0 30.5 14t1.5 33zM897 0h-595v50q0 21 14.5 35.5t35.5 14.5h50v50 q0 21 14.5 35.5t35.5 14.5h48v300h200v-300h47q21 0 35.5 -14.5t14.5 -35.5v-50h50q21 0 35.5 -14.5t14.5 -35.5v-50z" />
+<glyph unicode="&#xe224;" d="M900 800h300v-575q0 -10 -7.5 -17.5t-17.5 -7.5h-375v591l-300 300v84q0 10 7.5 17.5t17.5 7.5h375v-400zM1200 900h-200v200zM400 600h300v-575q0 -10 -7.5 -17.5t-17.5 -7.5h-650q-10 0 -17.5 7.5t-7.5 17.5v950q0 10 7.5 17.5t17.5 7.5h375v-400zM700 700h-200v200z " />
+<glyph unicode="&#xe225;" d="M484 1095h195q75 0 146 -32.5t124 -86t89.5 -122.5t48.5 -142q18 -14 35 -20q31 -10 64.5 6.5t43.5 48.5q10 34 -15 71q-19 27 -9 43q5 8 12.5 11t19 -1t23.5 -16q41 -44 39 -105q-3 -63 -46 -106.5t-104 -43.5h-62q-7 -55 -35 -117t-56 -100l-39 -234q-3 -20 -20 -34.5 t-38 -14.5h-100q-21 0 -33 14.5t-9 34.5l12 70q-49 -14 -91 -14h-195q-24 0 -65 8l-11 -64q-3 -20 -20 -34.5t-38 -14.5h-100q-21 0 -33 14.5t-9 34.5l26 157q-84 74 -128 175l-159 53q-19 7 -33 26t-14 40v50q0 21 14.5 35.5t35.5 14.5h124q11 87 56 166l-111 95 q-16 14 -12.5 23.5t24.5 9.5h203q116 101 250 101zM675 1000h-250q-10 0 -17.5 -7.5t-7.5 -17.5v-50q0 -10 7.5 -17.5t17.5 -7.5h250q10 0 17.5 7.5t7.5 17.5v50q0 10 -7.5 17.5t-17.5 7.5z" />
+<glyph unicode="&#xe226;" d="M641 900l423 247q19 8 42 2.5t37 -21.5l32 -38q14 -15 12.5 -36t-17.5 -34l-139 -120h-390zM50 1100h106q67 0 103 -17t66 -71l102 -212h823q21 0 35.5 -14.5t14.5 -35.5v-50q0 -21 -14 -40t-33 -26l-737 -132q-23 -4 -40 6t-26 25q-42 67 -100 67h-300q-62 0 -106 44 t-44 106v200q0 62 44 106t106 44zM173 928h-80q-19 0 -28 -14t-9 -35v-56q0 -51 42 -51h134q16 0 21.5 8t5.5 24q0 11 -16 45t-27 51q-18 28 -43 28zM550 727q-32 0 -54.5 -22.5t-22.5 -54.5t22.5 -54.5t54.5 -22.5t54.5 22.5t22.5 54.5t-22.5 54.5t-54.5 22.5zM130 389 l152 130q18 19 34 24t31 -3.5t24.5 -17.5t25.5 -28q28 -35 50.5 -51t48.5 -13l63 5l48 -179q13 -61 -3.5 -97.5t-67.5 -79.5l-80 -69q-47 -40 -109 -35.5t-103 51.5l-130 151q-40 47 -35.5 109.5t51.5 102.5zM380 377l-102 -88q-31 -27 2 -65l37 -43q13 -15 27.5 -19.5 t31.5 6.5l61 53q19 16 14 49q-2 20 -12 56t-17 45q-11 12 -19 14t-23 -8z" />
+<glyph unicode="&#xe227;" d="M625 1200h150q10 0 17.5 -7.5t7.5 -17.5v-109q79 -33 131 -87.5t53 -128.5q1 -46 -15 -84.5t-39 -61t-46 -38t-39 -21.5l-17 -6q6 0 15 -1.5t35 -9t50 -17.5t53 -30t50 -45t35.5 -64t14.5 -84q0 -59 -11.5 -105.5t-28.5 -76.5t-44 -51t-49.5 -31.5t-54.5 -16t-49.5 -6.5 t-43.5 -1v-75q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v75h-100v-75q0 -10 -7.5 -17.5t-17.5 -7.5h-150q-10 0 -17.5 7.5t-7.5 17.5v75h-175q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5h75v600h-75q-10 0 -17.5 7.5t-7.5 17.5v150 q0 10 7.5 17.5t17.5 7.5h175v75q0 10 7.5 17.5t17.5 7.5h150q10 0 17.5 -7.5t7.5 -17.5v-75h100v75q0 10 7.5 17.5t17.5 7.5zM400 900v-200h263q28 0 48.5 10.5t30 25t15 29t5.5 25.5l1 10q0 4 -0.5 11t-6 24t-15 30t-30 24t-48.5 11h-263zM400 500v-200h363q28 0 48.5 10.5 t30 25t15 29t5.5 25.5l1 10q0 4 -0.5 11t-6 24t-15 30t-30 24t-48.5 11h-363z" />
+<glyph unicode="&#xe230;" d="M212 1198h780q86 0 147 -61t61 -147v-416q0 -51 -18 -142.5t-36 -157.5l-18 -66q-29 -87 -93.5 -146.5t-146.5 -59.5h-572q-82 0 -147 59t-93 147q-8 28 -20 73t-32 143.5t-20 149.5v416q0 86 61 147t147 61zM600 1045q-70 0 -132.5 -11.5t-105.5 -30.5t-78.5 -41.5 t-57 -45t-36 -41t-20.5 -30.5l-6 -12l156 -243h560l156 243q-2 5 -6 12.5t-20 29.5t-36.5 42t-57 44.5t-79 42t-105 29.5t-132.5 12zM762 703h-157l195 261z" />
+<glyph unicode="&#xe231;" d="M475 1300h150q103 0 189 -86t86 -189v-500q0 -41 -42 -83t-83 -42h-450q-41 0 -83 42t-42 83v500q0 103 86 189t189 86zM700 300v-225q0 -21 -27 -48t-48 -27h-150q-21 0 -48 27t-27 48v225h300z" />
+<glyph unicode="&#xe232;" d="M475 1300h96q0 -150 89.5 -239.5t239.5 -89.5v-446q0 -41 -42 -83t-83 -42h-450q-41 0 -83 42t-42 83v500q0 103 86 189t189 86zM700 300v-225q0 -21 -27 -48t-48 -27h-150q-21 0 -48 27t-27 48v225h300z" />
+<glyph unicode="&#xe233;" d="M1294 767l-638 -283l-378 170l-78 -60v-224l100 -150v-199l-150 148l-150 -149v200l100 150v250q0 4 -0.5 10.5t0 9.5t1 8t3 8t6.5 6l47 40l-147 65l642 283zM1000 380l-350 -166l-350 166v147l350 -165l350 165v-147z" />
+<glyph unicode="&#xe234;" d="M250 800q62 0 106 -44t44 -106t-44 -106t-106 -44t-106 44t-44 106t44 106t106 44zM650 800q62 0 106 -44t44 -106t-44 -106t-106 -44t-106 44t-44 106t44 106t106 44zM1050 800q62 0 106 -44t44 -106t-44 -106t-106 -44t-106 44t-44 106t44 106t106 44z" />
+<glyph unicode="&#xe235;" d="M550 1100q62 0 106 -44t44 -106t-44 -106t-106 -44t-106 44t-44 106t44 106t106 44zM550 700q62 0 106 -44t44 -106t-44 -106t-106 -44t-106 44t-44 106t44 106t106 44zM550 300q62 0 106 -44t44 -106t-44 -106t-106 -44t-106 44t-44 106t44 106t106 44z" />
+<glyph unicode="&#xe236;" d="M125 1100h950q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-950q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5zM125 700h950q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-950q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5 t17.5 7.5zM125 300h950q10 0 17.5 -7.5t7.5 -17.5v-150q0 -10 -7.5 -17.5t-17.5 -7.5h-950q-10 0 -17.5 7.5t-7.5 17.5v150q0 10 7.5 17.5t17.5 7.5z" />
+<glyph unicode="&#xe237;" d="M350 1200h500q162 0 256 -93.5t94 -256.5v-500q0 -165 -93.5 -257.5t-256.5 -92.5h-500q-165 0 -257.5 92.5t-92.5 257.5v500q0 165 92.5 257.5t257.5 92.5zM900 1000h-600q-41 0 -70.5 -29.5t-29.5 -70.5v-600q0 -41 29.5 -70.5t70.5 -29.5h600q41 0 70.5 29.5 t29.5 70.5v600q0 41 -29.5 70.5t-70.5 29.5zM350 900h500q21 0 35.5 -14.5t14.5 -35.5v-300q0 -21 -14.5 -35.5t-35.5 -14.5h-500q-21 0 -35.5 14.5t-14.5 35.5v300q0 21 14.5 35.5t35.5 14.5zM400 800v-200h400v200h-400z" />
+<glyph unicode="&#xe238;" d="M150 1100h1000q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-50v-200h50q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-50v-200h50q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5t-35.5 -14.5h-50v-200h50q21 0 35.5 -14.5t14.5 -35.5t-14.5 -35.5 t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5h50v200h-50q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5h50v200h-50q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5h50v200h-50q-21 0 -35.5 14.5t-14.5 35.5t14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe239;" d="M650 1187q87 -67 118.5 -156t0 -178t-118.5 -155q-87 66 -118.5 155t0 178t118.5 156zM300 800q124 0 212 -88t88 -212q-124 0 -212 88t-88 212zM1000 800q0 -124 -88 -212t-212 -88q0 124 88 212t212 88zM300 500q124 0 212 -88t88 -212q-124 0 -212 88t-88 212z M1000 500q0 -124 -88 -212t-212 -88q0 124 88 212t212 88zM700 199v-144q0 -21 -14.5 -35.5t-35.5 -14.5t-35.5 14.5t-14.5 35.5v142q40 -4 43 -4q17 0 57 6z" />
+<glyph unicode="&#xe240;" d="M745 878l69 19q25 6 45 -12l298 -295q11 -11 15 -26.5t-2 -30.5q-5 -14 -18 -23.5t-28 -9.5h-8q1 0 1 -13q0 -29 -2 -56t-8.5 -62t-20 -63t-33 -53t-51 -39t-72.5 -14h-146q-184 0 -184 288q0 24 10 47q-20 4 -62 4t-63 -4q11 -24 11 -47q0 -288 -184 -288h-142 q-48 0 -84.5 21t-56 51t-32 71.5t-16 75t-3.5 68.5q0 13 2 13h-7q-15 0 -27.5 9.5t-18.5 23.5q-6 15 -2 30.5t15 25.5l298 296q20 18 46 11l76 -19q20 -5 30.5 -22.5t5.5 -37.5t-22.5 -31t-37.5 -5l-51 12l-182 -193h891l-182 193l-44 -12q-20 -5 -37.5 6t-22.5 31t6 37.5 t31 22.5z" />
+<glyph unicode="&#xe241;" d="M1200 900h-50q0 21 -4 37t-9.5 26.5t-18 17.5t-22 11t-28.5 5.5t-31 2t-37 0.5h-200v-850q0 -22 25 -34.5t50 -13.5l25 -2v-100h-400v100q4 0 11 0.5t24 3t30 7t24 15t11 24.5v850h-200q-25 0 -37 -0.5t-31 -2t-28.5 -5.5t-22 -11t-18 -17.5t-9.5 -26.5t-4 -37h-50v300 h1000v-300zM500 450h-25q0 15 -4 24.5t-9 14.5t-17 7.5t-20 3t-25 0.5h-100v-425q0 -11 12.5 -17.5t25.5 -7.5h12v-50h-200v50q50 0 50 25v425h-100q-17 0 -25 -0.5t-20 -3t-17 -7.5t-9 -14.5t-4 -24.5h-25v150h500v-150z" />
+<glyph unicode="&#xe242;" d="M1000 300v50q-25 0 -55 32q-14 14 -25 31t-16 27l-4 11l-289 747h-69l-300 -754q-18 -35 -39 -56q-9 -9 -24.5 -18.5t-26.5 -14.5l-11 -5v-50h273v50q-49 0 -78.5 21.5t-11.5 67.5l69 176h293l61 -166q13 -34 -3.5 -66.5t-55.5 -32.5v-50h312zM412 691l134 342l121 -342 h-255zM1100 150v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h1000q21 0 35.5 -14.5t14.5 -35.5z" />
+<glyph unicode="&#xe243;" d="M50 1200h1100q21 0 35.5 -14.5t14.5 -35.5v-1100q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v1100q0 21 14.5 35.5t35.5 14.5zM611 1118h-70q-13 0 -18 -12l-299 -753q-17 -32 -35 -51q-18 -18 -56 -34q-12 -5 -12 -18v-50q0 -8 5.5 -14t14.5 -6 h273q8 0 14 6t6 14v50q0 8 -6 14t-14 6q-55 0 -71 23q-10 14 0 39l63 163h266l57 -153q11 -31 -6 -55q-12 -17 -36 -17q-8 0 -14 -6t-6 -14v-50q0 -8 6 -14t14 -6h313q8 0 14 6t6 14v50q0 7 -5.5 13t-13.5 7q-17 0 -42 25q-25 27 -40 63h-1l-288 748q-5 12 -19 12zM639 611 h-197l103 264z" />
+<glyph unicode="&#xe244;" d="M1200 1100h-1200v100h1200v-100zM50 1000h400q21 0 35.5 -14.5t14.5 -35.5v-900q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v900q0 21 14.5 35.5t35.5 14.5zM650 1000h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400 q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM700 900v-300h300v300h-300z" />
+<glyph unicode="&#xe245;" d="M50 1200h400q21 0 35.5 -14.5t14.5 -35.5v-900q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v900q0 21 14.5 35.5t35.5 14.5zM650 700h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400 q0 21 14.5 35.5t35.5 14.5zM700 600v-300h300v300h-300zM1200 0h-1200v100h1200v-100z" />
+<glyph unicode="&#xe246;" d="M50 1000h400q21 0 35.5 -14.5t14.5 -35.5v-350h100v150q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5v-150h100v-100h-100v-150q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v150h-100v-350q0 -21 -14.5 -35.5t-35.5 -14.5h-400 q-21 0 -35.5 14.5t-14.5 35.5v800q0 21 14.5 35.5t35.5 14.5zM700 700v-300h300v300h-300z" />
+<glyph unicode="&#xe247;" d="M100 0h-100v1200h100v-1200zM250 1100h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM300 1000v-300h300v300h-300zM250 500h900q21 0 35.5 -14.5t14.5 -35.5v-400 q0 -21 -14.5 -35.5t-35.5 -14.5h-900q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe248;" d="M600 1100h150q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-150v-100h450q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-900q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5h350v100h-150q-21 0 -35.5 14.5 t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5h150v100h100v-100zM400 1000v-300h300v300h-300z" />
+<glyph unicode="&#xe249;" d="M1200 0h-100v1200h100v-1200zM550 1100h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM600 1000v-300h300v300h-300zM50 500h900q21 0 35.5 -14.5t14.5 -35.5v-400 q0 -21 -14.5 -35.5t-35.5 -14.5h-900q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5z" />
+<glyph unicode="&#xe250;" d="M865 565l-494 -494q-23 -23 -41 -23q-14 0 -22 13.5t-8 38.5v1000q0 25 8 38.5t22 13.5q18 0 41 -23l494 -494q14 -14 14 -35t-14 -35z" />
+<glyph unicode="&#xe251;" d="M335 635l494 494q29 29 50 20.5t21 -49.5v-1000q0 -41 -21 -49.5t-50 20.5l-494 494q-14 14 -14 35t14 35z" />
+<glyph unicode="&#xe252;" d="M100 900h1000q41 0 49.5 -21t-20.5 -50l-494 -494q-14 -14 -35 -14t-35 14l-494 494q-29 29 -20.5 50t49.5 21z" />
+<glyph unicode="&#xe253;" d="M635 865l494 -494q29 -29 20.5 -50t-49.5 -21h-1000q-41 0 -49.5 21t20.5 50l494 494q14 14 35 14t35 -14z" />
+<glyph unicode="&#xe254;" d="M700 741v-182l-692 -323v221l413 193l-413 193v221zM1200 0h-800v200h800v-200z" />
+<glyph unicode="&#xe255;" d="M1200 900h-200v-100h200v-100h-300v300h200v100h-200v100h300v-300zM0 700h50q0 21 4 37t9.5 26.5t18 17.5t22 11t28.5 5.5t31 2t37 0.5h100v-550q0 -22 -25 -34.5t-50 -13.5l-25 -2v-100h400v100q-4 0 -11 0.5t-24 3t-30 7t-24 15t-11 24.5v550h100q25 0 37 -0.5t31 -2 t28.5 -5.5t22 -11t18 -17.5t9.5 -26.5t4 -37h50v300h-800v-300z" />
+<glyph unicode="&#xe256;" d="M800 700h-50q0 21 -4 37t-9.5 26.5t-18 17.5t-22 11t-28.5 5.5t-31 2t-37 0.5h-100v-550q0 -22 25 -34.5t50 -14.5l25 -1v-100h-400v100q4 0 11 0.5t24 3t30 7t24 15t11 24.5v550h-100q-25 0 -37 -0.5t-31 -2t-28.5 -5.5t-22 -11t-18 -17.5t-9.5 -26.5t-4 -37h-50v300 h800v-300zM1100 200h-200v-100h200v-100h-300v300h200v100h-200v100h300v-300z" />
+<glyph unicode="&#xe257;" d="M701 1098h160q16 0 21 -11t-7 -23l-464 -464l464 -464q12 -12 7 -23t-21 -11h-160q-13 0 -23 9l-471 471q-7 8 -7 18t7 18l471 471q10 9 23 9z" />
+<glyph unicode="&#xe258;" d="M339 1098h160q13 0 23 -9l471 -471q7 -8 7 -18t-7 -18l-471 -471q-10 -9 -23 -9h-160q-16 0 -21 11t7 23l464 464l-464 464q-12 12 -7 23t21 11z" />
+<glyph unicode="&#xe259;" d="M1087 882q11 -5 11 -21v-160q0 -13 -9 -23l-471 -471q-8 -7 -18 -7t-18 7l-471 471q-9 10 -9 23v160q0 16 11 21t23 -7l464 -464l464 464q12 12 23 7z" />
+<glyph unicode="&#xe260;" d="M618 993l471 -471q9 -10 9 -23v-160q0 -16 -11 -21t-23 7l-464 464l-464 -464q-12 -12 -23 -7t-11 21v160q0 13 9 23l471 471q8 7 18 7t18 -7z" />
+<glyph unicode="&#xf8ff;" d="M1000 1200q0 -124 -88 -212t-212 -88q0 124 88 212t212 88zM450 1000h100q21 0 40 -14t26 -33l79 -194q5 1 16 3q34 6 54 9.5t60 7t65.5 1t61 -10t56.5 -23t42.5 -42t29 -64t5 -92t-19.5 -121.5q-1 -7 -3 -19.5t-11 -50t-20.5 -73t-32.5 -81.5t-46.5 -83t-64 -70 t-82.5 -50q-13 -5 -42 -5t-65.5 2.5t-47.5 2.5q-14 0 -49.5 -3.5t-63 -3.5t-43.5 7q-57 25 -104.5 78.5t-75 111.5t-46.5 112t-26 90l-7 35q-15 63 -18 115t4.5 88.5t26 64t39.5 43.5t52 25.5t58.5 13t62.5 2t59.5 -4.5t55.5 -8l-147 192q-12 18 -5.5 30t27.5 12z" />
+<glyph unicode="&#x1f511;" d="M250 1200h600q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-150v-500l-255 -178q-19 -9 -32 -1t-13 29v650h-150q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5zM400 1100v-100h300v100h-300z" />
+<glyph unicode="&#x1f6aa;" d="M250 1200h750q39 0 69.5 -40.5t30.5 -84.5v-933l-700 -117v950l600 125h-700v-1000h-100v1025q0 23 15.5 49t34.5 26zM500 525v-100l100 20v100z" />
+</font>
+</defs></svg> 

+ 0 - 0
src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.ttf


+ 0 - 0
src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.woff


+ 0 - 0
src/main/resources/static/api/app-api/styles/bootstrap/fonts/glyphicons-halflings-regular.woff2


Fichier diff supprimé car celui-ci est trop grand
+ 5 - 0
src/main/resources/static/api/app-api/styles/bootstrap/js/bootstrap.min.js


Fichier diff supprimé car celui-ci est trop grand
+ 1 - 0
src/main/resources/static/api/app-api/styles/jquery/jquery-1.9.1.min.js


+ 35 - 0
src/main/resources/static/api/app-api/upload.html

@@ -0,0 +1,35 @@
+<!DOCTYPE html>
+<html lang="zh-cn">
+<head>
+    <meta name="renderer" content="webkit"/>
+    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>
+    <script src="/styles/jquery/jquery-1.9.1.min.js"></script>
+</head>
+<body>
+<form id="uploadForm" enctype="multipart/form-data">
+    <input type="file" name="file"/>
+    <input type="button" value="提交" onclick="doUpload()"/>
+</form>
+<script type="text/javascript">
+    function doUpload() {
+        var formData = new FormData($('#uploadForm')[0]);
+        formData.append('examRecordId', '101262');
+        $.ajax({
+            url: 'http://localhost:8090/api/v1/exam/offline/paper/answer/upload',
+            type: "POST",
+            data: formData,
+            headers: {
+                'key': 'U_S_109_53308',
+                'token': '9e3eb5be1d2a4a9e9ffab4341af7b987'
+            },
+            processData: false,
+            contentType: false,
+            success: function (data) {
+                console.log(data);
+            }
+        });
+    }
+</script>
+</body>
+</html>

+ 191 - 0
src/main/resources/templates/deviceRecord/list.ftl

@@ -0,0 +1,191 @@
+<#assign base=request.contextPath>
+<!DOCTYPE html>
+<html lang="zh-cn">
+<head>
+    <title>设备访问记录</title>
+    <meta name="renderer" content="webkit"/>
+    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>
+    <link href="${base}/api/app-api/styles/bootstrap/css/bootstrap.min.css" rel="stylesheet"/>
+    <style type="text/css">
+        input {
+            margin-bottom: 10px;
+        }
+
+        span {
+            margin-right: 15px;
+        }
+
+        .pagination > li > a {
+            border: 0;
+        }
+    </style>
+</head>
+<body>
+<h3 style="margin-left: 12px">设备访问记录</h3>
+<div class="panel panel-default" style="padding: 10px 10px 0px 10px">
+    <div class="btn-group">
+        <input type="text" class="form-control" id="url" placeholder="访问地址"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="system" placeholder="系统标识"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="deviceId" placeholder="设备编号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="netType" placeholder="网络类型"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="brand" placeholder="设备品牌"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="model" placeholder="型号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="sysVersion" placeholder="系统版本号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="appVersion" placeholder="APP版本号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="appCode" placeholder="APP更新号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="patchCode" placeholder="APP补丁号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="account" placeholder="来源账号"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="loginKey" placeholder="KEY"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="loginToken" placeholder="TOKEN"/>
+    </div>
+    <div class="btn-group">
+        <input type="text" class="form-control" id="ip" placeholder="IP"/>
+    </div>
+    <div class="btn-group">
+        <a href="#" class="btn btn-primary" onclick="doSearch()" style="margin-bottom: 10px"> 搜索</a>
+    </div>
+    <table class="table">
+        <thead>
+        <tr>
+            <th style="width: 150px">操作时间</th>
+            <th>操作内容</th>
+        </tr>
+        </thead>
+        <tbody id="contentTable"></tbody>
+    </table>
+    <div style="text-align: right;margin: 0;padding: 0">
+        <ul id="contentPage" class="pagination"></ul>
+    </div>
+</div>
+
+<script src="${base}/api/app-api/styles/jquery/jquery-1.9.1.min.js"></script>
+<script src="${base}/api/app-api/page.js?v=20180801"></script>
+<script type="text/javascript">
+    var basePath = '${base!}/api/app-api';
+    var curPageSize = 10;
+
+    $(function () {
+        loadData(1);
+    });
+
+    function doSearch() {
+        loadData(1);
+    }
+
+    function loadData(curPageNo) {
+        var params = {};
+        params.url = $('#url').val();
+        params.system = $('#system').val();
+        params.deviceId = $('#deviceId').val();
+        params.netType = $('#netType').val();
+        params.brand = $('#brand').val();
+        params.model = $('#model').val();
+        params.sysVersion = $('#sysVersion').val();
+        params.appVersion = $('#appVersion').val();
+        params.appCode = $('#appCode').val();
+        params.patchCode = $('#patchCode').val();
+        params.account = $('#account').val();
+        params.loginKey = $('#loginKey').val();
+        params.loginToken = $('#loginToken').val();
+        params.ip = $('#ip').val();
+        params.pageSize = curPageSize;
+        params.pageNo = curPageNo;
+
+        $("#contentTable").empty();
+        $("#contentPage").empty();
+        $.ajax({
+            url: basePath + '/device/record/list?key=${key!}&token=${token!}',
+            contentType: "application/json; charset=UTF-8",
+            dataType: "json",
+            type: "POST",
+            data: JSON.stringify(params),
+            success: function (response) {
+                if (!response || response.code != '200') {
+                    console.error(response);
+                    return;
+                }
+                var page = response.data;
+                var list = render(page.content);
+                $("#contentTable").html(list);
+
+                $("#contentPage").createPage({
+                    totalElements: page.totalElements,
+                    totalPages: page.totalPages,
+                    current: curPageNo,
+                    backFn: function (current) {
+                        loadData(current);
+                    }
+                });
+            }
+        });
+    }
+
+    function render(list) {
+        var html = [];
+        if (!list || list.length == 0) {
+            return html.join('');
+        }
+        for (var n = 0; n < list.length; n++) {
+            var obj = list[n];
+            if (n % 2 == 0) {
+                html.push('<tr>');
+            } else {
+                html.push('<tr style="background-color:#eef1f6">');
+            }
+            html.push('<td style="vertical-align: middle;text-align: center;">');
+            html.push(emptyStr(obj.createDate) + '</br>' + emptyStr(obj.ip));
+            html.push('</td>');
+            html.push('<td>');
+            html.push('<span>访问地址:' + emptyStr(obj.url) + '</span>');
+            html.push('<span>KEY:' + emptyStr(obj.loginKey) + '</span>');
+            html.push('<span>TOKEN:' + emptyStr(obj.loginToken) + '</span></br>');
+            html.push('<span>来源账号:' + emptyStr(obj.account) + '</span>');
+            html.push('<span>系统标识:' + emptyStr(obj.system) + '</span>');
+            html.push('<span>设备编号:' + emptyStr(obj.deviceId) + '</span>');
+            html.push('<span>网络类型:' + emptyStr(obj.netType) + '</span>');
+            html.push('<span>设备品牌:' + emptyStr(obj.brand) + '</span>');
+            html.push('<span>型号:' + emptyStr(obj.model) + '</span>');
+            html.push('<span>系统版本号:' + emptyStr(obj.sysVersion) + '</span>');
+            html.push('<span>APP版本号:' + emptyStr(obj.appVersion) + '</span>');
+            html.push('<span>APP更新号:' + emptyStr(obj.appCode) + '</span>');
+            html.push('<span>APP补丁号:' + emptyStr(obj.patchCode) + '</span>');
+            html.push('</td>');
+            html.push('</tr>');
+        }
+        return html.join('');
+    }
+
+    function emptyStr(str) {
+        if (!str || str == null) {
+            return "-";
+        }
+        return str;
+    }
+</script>
+</body>
+</html>

+ 37 - 0
src/main/resources/templates/error.ftl

@@ -0,0 +1,37 @@
+<#assign base=request.contextPath>
+<!DOCTYPE html>
+<html lang="zh-cn">
+<head>
+    <title>${status!}</title>
+    <meta name="renderer" content="webkit"/>
+    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>
+</head>
+<body>
+<div style="padding-left: 10px;">
+    <h1>${status!}</h1>
+    <h3>
+        <#if status??>
+            <#if status==401>
+                未授权,访问由服务器拒绝!
+            <#elseif status==403>
+                禁止访问,无访问权限!
+            <#elseif status==404>
+                抱歉,页面不存在!
+            <#else>
+                系统错误,请与管理员联系!
+            </#if>
+        </#if>
+    </h3>
+    <p><a href="${base}/api/app-api">返回首页</a></p>
+    <#if status??>
+        <#if status==500>
+            <p><i>${errMsg!}</i></p>
+            <p><i>${error!}</i></p>
+            <p><i>${exception!}</i></p>
+            <p><i>${message!}</i></p>
+        </#if>
+    </#if>
+</div>
+</body>
+</html>

+ 15 - 0
src/main/resources/templates/index.ftl

@@ -0,0 +1,15 @@
+<#assign base=request.contextPath>
+<!DOCTYPE html>
+<html lang="zh-cn">
+<head>
+    <title>首页</title>
+    <meta name="renderer" content="webkit"/>
+    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
+    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>
+</head>
+<body>
+<script>
+    window.location.href = '${base}/doc.html';
+</script>
+</body>
+</html>

+ 34 - 0
src/test/java/cn/com/qmth/examcloud/app/ServiceTest.java

@@ -0,0 +1,34 @@
+package cn.com.qmth.examcloud.app;
+
+import cn.com.qmth.examcloud.app.core.utils.JsonMapper;
+import cn.com.qmth.examcloud.app.model.LoginInfo;
+import cn.com.qmth.examcloud.app.model.UserInfo;
+import cn.com.qmth.examcloud.app.service.RedisService;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.test.context.SpringBootTest;
+import org.springframework.test.context.junit4.SpringRunner;
+
+import static cn.com.qmth.examcloud.app.model.Constants.APP_SESSION_USER_KEY_PREFIX;
+
+@RunWith(SpringRunner.class)
+@SpringBootTest
+public class ServiceTest {
+    @Autowired
+    private RedisService redisService;
+
+    @Test
+    public void demo() throws Exception {
+        String key = "U_C_109_18809";
+        redisService.setObj(key, new UserInfo(), 30);
+
+        UserInfo user = redisService.getObj(key, UserInfo.class);
+        System.out.println(new JsonMapper().toJson(user));
+
+        key = APP_SESSION_USER_KEY_PREFIX + key;
+        LoginInfo loginInfo = redisService.fromJson(key, LoginInfo.class);
+        System.out.println(new JsonMapper().toJson(loginInfo));
+    }
+
+}

+ 16 - 0
src/test/java/cn/com/qmth/examcloud/app/SimpleTest.java

@@ -0,0 +1,16 @@
+package cn.com.qmth.examcloud.app;
+
+import cn.com.qmth.examcloud.app.core.utils.DateUtils;
+import org.junit.Test;
+
+import java.time.Duration;
+
+public class SimpleTest {
+
+    @Test
+    public void demo() throws Exception {
+        System.out.println(Duration.ofSeconds(1));
+        System.out.println(DateUtils.formatLongDate("1533208461728"));
+    }
+
+}

Certains fichiers n'ont pas été affichés car il y a eu trop de fichiers modifiés dans ce diff