├── .gitignore
├── .mvn
└── wrapper
│ ├── maven-wrapper.jar
│ └── maven-wrapper.properties
├── README.md
├── demo.png
├── demo1.jpg
├── demo2.jpg
├── demo3.jpg
├── demo4.png
├── dev
├── Dockerfile
└── docker-compose.yml
├── dmo3.jpg
├── golang.conf
├── mvnw
├── mvnw.cmd
├── pom.xml
├── scripts
└── init.sh
└── src
├── main
├── java
│ └── com
│ │ └── bektz
│ │ └── dataplatformsoar
│ │ ├── DataPlatformSoarApplication.java
│ │ ├── config.java
│ │ ├── configs
│ │ ├── MyWebSocketHandler.java
│ │ ├── ParamValidAop.java
│ │ ├── SoarConfiguration.java
│ │ └── WebSocketConfig.java
│ │ ├── constants
│ │ ├── Common.java
│ │ ├── JdbcConstants.java
│ │ ├── QueryStatus.java
│ │ └── QueryType.java
│ │ ├── controller
│ │ ├── PageController.java
│ │ ├── SchemaController.java
│ │ ├── SoarController.java
│ │ └── handler
│ │ │ └── ControllerExceptionHandleAdvice.java
│ │ ├── exception
│ │ ├── BektzClientException.java
│ │ ├── BektzException.java
│ │ ├── BektzServerException.java
│ │ └── SqlParserException.java
│ │ ├── repository
│ │ ├── SecretColumnRepository.java
│ │ └── domain
│ │ │ └── SecretColumn.java
│ │ ├── req
│ │ ├── SchemaReq.java
│ │ └── SqlVerifyReq.java
│ │ ├── resp
│ │ ├── GenericResponse.java
│ │ ├── JdbcResultResp.java
│ │ ├── SchemaResp.java
│ │ └── SqlVerifyResp.java
│ │ ├── service
│ │ ├── SchemaService.java
│ │ ├── SoarService.java
│ │ └── jdbc
│ │ │ ├── DataSourceManager.java
│ │ │ └── JdbcTemplate.java
│ │ └── sqlparser
│ │ ├── ColumnItem.java
│ │ ├── CommonFactory.java
│ │ ├── SqlParserService.java
│ │ └── druid
│ │ ├── AbstractDruidSqlParSer.java
│ │ ├── DruidParserHasAsterisk.java
│ │ ├── DruidParserNonAsterisk.java
│ │ ├── DruidSqlParser.java
│ │ └── visitor
│ │ └── DefineMySqlSchemaStatVisitor.java
└── resources
│ ├── application.yml
│ ├── soar.yml
│ ├── static
│ ├── css
│ │ └── prism.css
│ ├── images
│ │ └── face
│ │ │ ├── 0.gif
│ │ │ ├── 1.gif
│ │ │ ├── 10.gif
│ │ │ ├── 11.gif
│ │ │ ├── 12.gif
│ │ │ ├── 13.gif
│ │ │ ├── 14.gif
│ │ │ ├── 15.gif
│ │ │ ├── 16.gif
│ │ │ ├── 17.gif
│ │ │ ├── 18.gif
│ │ │ ├── 19.gif
│ │ │ ├── 2.gif
│ │ │ ├── 20.gif
│ │ │ ├── 21.gif
│ │ │ ├── 22.gif
│ │ │ ├── 23.gif
│ │ │ ├── 24.gif
│ │ │ ├── 25.gif
│ │ │ ├── 26.gif
│ │ │ ├── 27.gif
│ │ │ ├── 28.gif
│ │ │ ├── 29.gif
│ │ │ ├── 3.gif
│ │ │ ├── 30.gif
│ │ │ ├── 31.gif
│ │ │ ├── 32.gif
│ │ │ ├── 33.gif
│ │ │ ├── 34.gif
│ │ │ ├── 35.gif
│ │ │ ├── 36.gif
│ │ │ ├── 37.gif
│ │ │ ├── 38.gif
│ │ │ ├── 39.gif
│ │ │ ├── 4.gif
│ │ │ ├── 40.gif
│ │ │ ├── 41.gif
│ │ │ ├── 42.gif
│ │ │ ├── 43.gif
│ │ │ ├── 44.gif
│ │ │ ├── 45.gif
│ │ │ ├── 46.gif
│ │ │ ├── 47.gif
│ │ │ ├── 48.gif
│ │ │ ├── 49.gif
│ │ │ ├── 5.gif
│ │ │ ├── 50.gif
│ │ │ ├── 51.gif
│ │ │ ├── 52.gif
│ │ │ ├── 53.gif
│ │ │ ├── 54.gif
│ │ │ ├── 55.gif
│ │ │ ├── 56.gif
│ │ │ ├── 57.gif
│ │ │ ├── 58.gif
│ │ │ ├── 59.gif
│ │ │ ├── 6.gif
│ │ │ ├── 60.gif
│ │ │ ├── 61.gif
│ │ │ ├── 62.gif
│ │ │ ├── 63.gif
│ │ │ ├── 64.gif
│ │ │ ├── 65.gif
│ │ │ ├── 66.gif
│ │ │ ├── 67.gif
│ │ │ ├── 68.gif
│ │ │ ├── 69.gif
│ │ │ ├── 7.gif
│ │ │ ├── 70.gif
│ │ │ ├── 71.gif
│ │ │ ├── 8.gif
│ │ │ └── 9.gif
│ ├── jquery.js
│ ├── js
│ │ ├── form.js
│ │ ├── lay.js
│ │ ├── prism.js
│ │ └── websocket.js
│ └── layui
│ │ ├── css
│ │ ├── layui.css
│ │ ├── layui.mobile.css
│ │ └── modules
│ │ │ ├── code.css
│ │ │ ├── laydate
│ │ │ └── default
│ │ │ │ └── laydate.css
│ │ │ └── layer
│ │ │ └── default
│ │ │ ├── icon-ext.png
│ │ │ ├── icon.png
│ │ │ ├── layer.css
│ │ │ ├── loading-0.gif
│ │ │ ├── loading-1.gif
│ │ │ └── loading-2.gif
│ │ ├── font
│ │ ├── iconfont.eot
│ │ ├── iconfont.svg
│ │ ├── iconfont.ttf
│ │ └── iconfont.woff
│ │ ├── images
│ │ └── face
│ │ │ ├── 0.gif
│ │ │ ├── 1.gif
│ │ │ ├── 10.gif
│ │ │ ├── 11.gif
│ │ │ ├── 12.gif
│ │ │ ├── 13.gif
│ │ │ ├── 14.gif
│ │ │ ├── 15.gif
│ │ │ ├── 16.gif
│ │ │ ├── 17.gif
│ │ │ ├── 18.gif
│ │ │ ├── 19.gif
│ │ │ ├── 2.gif
│ │ │ ├── 20.gif
│ │ │ ├── 21.gif
│ │ │ ├── 22.gif
│ │ │ ├── 23.gif
│ │ │ ├── 24.gif
│ │ │ ├── 25.gif
│ │ │ ├── 26.gif
│ │ │ ├── 27.gif
│ │ │ ├── 28.gif
│ │ │ ├── 29.gif
│ │ │ ├── 3.gif
│ │ │ ├── 30.gif
│ │ │ ├── 31.gif
│ │ │ ├── 32.gif
│ │ │ ├── 33.gif
│ │ │ ├── 34.gif
│ │ │ ├── 35.gif
│ │ │ ├── 36.gif
│ │ │ ├── 37.gif
│ │ │ ├── 38.gif
│ │ │ ├── 39.gif
│ │ │ ├── 4.gif
│ │ │ ├── 40.gif
│ │ │ ├── 41.gif
│ │ │ ├── 42.gif
│ │ │ ├── 43.gif
│ │ │ ├── 44.gif
│ │ │ ├── 45.gif
│ │ │ ├── 46.gif
│ │ │ ├── 47.gif
│ │ │ ├── 48.gif
│ │ │ ├── 49.gif
│ │ │ ├── 5.gif
│ │ │ ├── 50.gif
│ │ │ ├── 51.gif
│ │ │ ├── 52.gif
│ │ │ ├── 53.gif
│ │ │ ├── 54.gif
│ │ │ ├── 55.gif
│ │ │ ├── 56.gif
│ │ │ ├── 57.gif
│ │ │ ├── 58.gif
│ │ │ ├── 59.gif
│ │ │ ├── 6.gif
│ │ │ ├── 60.gif
│ │ │ ├── 61.gif
│ │ │ ├── 62.gif
│ │ │ ├── 63.gif
│ │ │ ├── 64.gif
│ │ │ ├── 65.gif
│ │ │ ├── 66.gif
│ │ │ ├── 67.gif
│ │ │ ├── 68.gif
│ │ │ ├── 69.gif
│ │ │ ├── 7.gif
│ │ │ ├── 70.gif
│ │ │ ├── 71.gif
│ │ │ ├── 8.gif
│ │ │ └── 9.gif
│ │ ├── lay
│ │ └── modules
│ │ │ ├── carousel.js
│ │ │ ├── code.js
│ │ │ ├── colorpicker.js
│ │ │ ├── element.js
│ │ │ ├── flow.js
│ │ │ ├── form.js
│ │ │ ├── jquery.js
│ │ │ ├── laydate.js
│ │ │ ├── layedit.js
│ │ │ ├── layer.js
│ │ │ ├── laypage.js
│ │ │ ├── laytpl.js
│ │ │ ├── mobile.js
│ │ │ ├── rate.js
│ │ │ ├── slider.js
│ │ │ ├── table.js
│ │ │ ├── tree.js
│ │ │ ├── upload.js
│ │ │ └── util.js
│ │ ├── layui.all.js
│ │ └── layui.js
│ └── templates
│ ├── index.html
│ └── schema.html
└── test
└── java
└── com
└── bektz
└── dataplatformsoar
├── DataPlatformSoarApplicationTests.java
├── MysqlTest.java
├── SqlParserTest.java
├── service
└── SoarServiceTest.java
└── sqlparser
└── DruidSqlParserTest.java
/.gitignore:
--------------------------------------------------------------------------------
1 | /target/
2 | !.mvn/wrapper/maven-wrapper.jar
3 |
4 | ### STS ###
5 | .apt_generated
6 | .classpath
7 | .factorypath
8 | .project
9 | .settings
10 | .springBeans
11 | .sts4-cache
12 |
13 | ### IntelliJ IDEA ###
14 | .idea
15 | *.iws
16 | *.iml
17 | *.ipr
18 |
19 | ### NetBeans ###
20 | /nbproject/private/
21 | /build/
22 | /nbbuild/
23 | /dist/
24 | /nbdist/
25 | /.nb-gradle/
--------------------------------------------------------------------------------
/.mvn/wrapper/maven-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/.mvn/wrapper/maven-wrapper.jar
--------------------------------------------------------------------------------
/.mvn/wrapper/maven-wrapper.properties:
--------------------------------------------------------------------------------
1 | distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.5.4/apache-maven-3.5.4-bin.zip
2 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # data-platform-soar
2 |
3 | ## 介绍
4 | 基于小米开源工具soar的java应用(sql审核平台),本项目是一个简单的spring boot应用,前端是采用前端开源框架layui,前端与后端的通信采用的是websocket方式
5 | ```txt
6 | 1.可以动态添加数据库,对指定表进行数据查询(没有对查询数据量没有限制)
7 | 2.集成soar,可以对平日的sql进行审核分析,规范sql的编写
8 | 3.通过druid对sql进行解析,可以对查询表做屏蔽敏感字段(需要完善)
9 | ```
10 |
11 | ### 运行
12 |
13 | #### 1. 推荐方式
14 | ``` shell
15 | 1.请参考soar的官方教程安装soar,并将soar的路径加入环境变量
16 |
17 | 2.进入项目主目录
18 |
19 | 3.mvn package -Dmaven.test.skip=true dependency:copy
20 |
21 | 4.cd dev
22 |
23 | 5.nohup java -jar data-soar.jar &
24 |
25 | ```
26 | ``` html
27 | 通过 http://localhost:8080 访问
28 | ```
29 |
30 | #### 2. docker运行
31 |
32 | ``` shell
33 | 1.进入项目主目录
34 | 2.mvn package dependency:copy
35 | 3.wget http://golangtc.com/static/go/go1.10.3.linux-amd64.tar.gz
36 | 4.docker build -f dev/Dockerfile -t boot-soar:latest --rm=true .
37 | 5.docker run
38 | ```
39 | ``` html
40 | 通过 http://localhost:8080 访问
41 | ```
42 |
43 | ### DEMO
44 | 
45 | 
46 | 
47 | 
48 |
--------------------------------------------------------------------------------
/demo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/demo.png
--------------------------------------------------------------------------------
/demo1.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/demo1.jpg
--------------------------------------------------------------------------------
/demo2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/demo2.jpg
--------------------------------------------------------------------------------
/demo3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/demo3.jpg
--------------------------------------------------------------------------------
/demo4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/demo4.png
--------------------------------------------------------------------------------
/dev/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM centos
2 | MAINTAINER chenlang@lattebank.com
3 |
4 | RUN yum install -y \
5 | java-1.8.0-openjdk \
6 | java-1.8.0-openjdk-devel \
7 | wget \
8 | git
9 |
10 | ADD go1.10.3.linux-amd64.tar.gz /root
11 | ADD golang.conf /root/golang.conf
12 | RUN mv /root/go /usr/local/
13 | RUN cat /root/golang.conf >> /etc/profile
14 | RUN echo "source /etc/profile" >> /root/.bashrc
15 | RUN mkdir -p /home/golang
16 |
17 | ENV TZ Asia/Shanghai
18 | ENV JAVA_HOME /etc/alternatives/jre
19 | ENV SERVER_PORT 8080
20 | ENV JMX_PORT 1099
21 | ENV JAVA_OPTS "-Xmx2G -Xms500M"
22 | ENV APP_NAME data-soar
23 | ENV SPRING_PROFILES_ACTIVE dev
24 |
25 | RUN mkdir /tmp/deploy/
26 | COPY dev/data-soar.jar /tmp/deploy/app.jar
27 | RUN go get -d github.com/XiaoMi/soar
28 | RUN cd ${GOPATH}/src/github.com/XiaoMi/soar && make
29 |
30 | EXPOSE 8080 1099
31 |
32 | ENTRYPOINT exec java $JAVA_OPTS -jar /tmp/deploy/app.jar
--------------------------------------------------------------------------------
/dev/docker-compose.yml:
--------------------------------------------------------------------------------
1 | version: '2'
2 | services:
3 | mysql:
4 | image: mariadb:latest
5 | container_name: mysql_service
6 | volumes:
7 | - ./data/mysql/data:/var/lib/mysql
8 | ports:
9 | - 3306:3306
10 | environment:
11 | - MYSQL_ROOT_PASSWORD=root
12 | data-platform-soar:
13 | build: .
14 | ports:
15 | - "8080:8080"
16 | networks:
17 | - local
18 | links:
19 | - "mysql"
--------------------------------------------------------------------------------
/dmo3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/beiketianzhuang/data-platform-soar/47e0d4f06d05b142cb57ef34891ad545af408427/dmo3.jpg
--------------------------------------------------------------------------------
/golang.conf:
--------------------------------------------------------------------------------
1 | export GOROOT=/usr/local/go
2 | export GOBIN=$GOROOT/bin
3 | export PATH=$PATH:$GOBIN
4 | export GOPATH=/home/golang
--------------------------------------------------------------------------------
/mvnw.cmd:
--------------------------------------------------------------------------------
1 | @REM ----------------------------------------------------------------------------
2 | @REM Licensed to the Apache Software Foundation (ASF) under one
3 | @REM or more contributor license agreements. See the NOTICE file
4 | @REM distributed with this work for additional information
5 | @REM regarding copyright ownership. The ASF licenses this file
6 | @REM to you under the Apache License, Version 2.0 (the
7 | @REM "License"); you may not use this file except in compliance
8 | @REM with the License. You may obtain a copy of the License at
9 | @REM
10 | @REM http://www.apache.org/licenses/LICENSE-2.0
11 | @REM
12 | @REM Unless required by applicable law or agreed to in writing,
13 | @REM software distributed under the License is distributed on an
14 | @REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 | @REM KIND, either express or implied. See the License for the
16 | @REM specific language governing permissions and limitations
17 | @REM under the License.
18 | @REM ----------------------------------------------------------------------------
19 |
20 | @REM ----------------------------------------------------------------------------
21 | @REM Maven2 Start Up Batch script
22 | @REM
23 | @REM Required ENV vars:
24 | @REM JAVA_HOME - location of a JDK home dir
25 | @REM
26 | @REM Optional ENV vars
27 | @REM M2_HOME - location of maven2's installed home dir
28 | @REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands
29 | @REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a key stroke before ending
30 | @REM MAVEN_OPTS - parameters passed to the Java VM when running Maven
31 | @REM e.g. to debug Maven itself, use
32 | @REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
33 | @REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files
34 | @REM ----------------------------------------------------------------------------
35 |
36 | @REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on'
37 | @echo off
38 | @REM set title of command window
39 | title %0
40 | @REM enable echoing my setting MAVEN_BATCH_ECHO to 'on'
41 | @if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO%
42 |
43 | @REM set %HOME% to equivalent of $HOME
44 | if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%")
45 |
46 | @REM Execute a user defined script before this one
47 | if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre
48 | @REM check for pre script, once with legacy .bat ending and once with .cmd ending
49 | if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat"
50 | if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd"
51 | :skipRcPre
52 |
53 | @setlocal
54 |
55 | set ERROR_CODE=0
56 |
57 | @REM To isolate internal variables from possible post scripts, we use another setlocal
58 | @setlocal
59 |
60 | @REM ==== START VALIDATION ====
61 | if not "%JAVA_HOME%" == "" goto OkJHome
62 |
63 | echo.
64 | echo Error: JAVA_HOME not found in your environment. >&2
65 | echo Please set the JAVA_HOME variable in your environment to match the >&2
66 | echo location of your Java installation. >&2
67 | echo.
68 | goto error
69 |
70 | :OkJHome
71 | if exist "%JAVA_HOME%\bin\java.exe" goto init
72 |
73 | echo.
74 | echo Error: JAVA_HOME is set to an invalid directory. >&2
75 | echo JAVA_HOME = "%JAVA_HOME%" >&2
76 | echo Please set the JAVA_HOME variable in your environment to match the >&2
77 | echo location of your Java installation. >&2
78 | echo.
79 | goto error
80 |
81 | @REM ==== END VALIDATION ====
82 |
83 | :init
84 |
85 | @REM Find the project base dir, i.e. the directory that contains the folder ".mvn".
86 | @REM Fallback to current working directory if not found.
87 |
88 | set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR%
89 | IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir
90 |
91 | set EXEC_DIR=%CD%
92 | set WDIR=%EXEC_DIR%
93 | :findBaseDir
94 | IF EXIST "%WDIR%"\.mvn goto baseDirFound
95 | cd ..
96 | IF "%WDIR%"=="%CD%" goto baseDirNotFound
97 | set WDIR=%CD%
98 | goto findBaseDir
99 |
100 | :baseDirFound
101 | set MAVEN_PROJECTBASEDIR=%WDIR%
102 | cd "%EXEC_DIR%"
103 | goto endDetectBaseDir
104 |
105 | :baseDirNotFound
106 | set MAVEN_PROJECTBASEDIR=%EXEC_DIR%
107 | cd "%EXEC_DIR%"
108 |
109 | :endDetectBaseDir
110 |
111 | IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig
112 |
113 | @setlocal EnableExtensions EnableDelayedExpansion
114 | for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a
115 | @endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS%
116 |
117 | :endReadAdditionalConfig
118 |
119 | SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe"
120 | set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
121 | set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
122 |
123 | set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.4.2/maven-wrapper-0.4.2.jar"
124 | FOR /F "tokens=1,2 delims==" %%A IN (%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties) DO (
125 | IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
126 | )
127 |
128 | @REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
129 | @REM This allows using the maven wrapper in projects that prohibit checking in binary data.
130 | if exist %WRAPPER_JAR% (
131 | echo Found %WRAPPER_JAR%
132 | ) else (
133 | echo Couldn't find %WRAPPER_JAR%, downloading it ...
134 | echo Downloading from: %DOWNLOAD_URL%
135 | powershell -Command "(New-Object Net.WebClient).DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"
136 | echo Finished downloading %WRAPPER_JAR%
137 | )
138 | @REM End of extension
139 |
140 | %MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %*
141 | if ERRORLEVEL 1 goto error
142 | goto end
143 |
144 | :error
145 | set ERROR_CODE=1
146 |
147 | :end
148 | @endlocal & set ERROR_CODE=%ERROR_CODE%
149 |
150 | if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost
151 | @REM check for post script, once with legacy .bat ending and once with .cmd ending
152 | if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat"
153 | if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd"
154 | :skipRcPost
155 |
156 | @REM pause the script if MAVEN_BATCH_PAUSE is set to 'on'
157 | if "%MAVEN_BATCH_PAUSE%" == "on" pause
158 |
159 | if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE%
160 |
161 | exit /B %ERROR_CODE%
162 |
--------------------------------------------------------------------------------
/pom.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 | 4.0.0
5 |
6 | com.bektz
7 | data-platform-soar
8 | 0.0.1-SNAPSHOT
9 | jar
10 |
11 | data-platform-soar
12 | Demo project for Spring Boot
13 |
14 |
15 | org.springframework.boot
16 | spring-boot-starter-parent
17 | 2.1.0.RELEASE
18 |
19 |
20 |
21 |
22 | UTF-8
23 | UTF-8
24 | 1.8
25 |
26 |
27 |
28 |
29 | org.apache.commons
30 | commons-lang3
31 | 3.4
32 |
33 |
34 | org.springframework.boot
35 | spring-boot-starter
36 |
37 |
38 | org.springframework.boot
39 | spring-boot-starter-test
40 | test
41 |
42 |
43 | org.springframework.boot
44 | spring-boot-starter-web
45 |
46 |
47 | org.springframework.boot
48 | spring-boot-starter-websocket
49 |
50 |
51 | org.springframework.boot
52 | spring-boot-starter-data-jpa
53 |
54 |
55 | org.projectlombok
56 | lombok
57 | 1.18.2
58 |
59 |
60 | com.github.jsqlparser
61 | jsqlparser
62 | 1.2
63 |
64 |
65 | org.junit.jupiter
66 | junit-jupiter-api
67 | RELEASE
68 |
69 |
70 | org.springframework.boot
71 | spring-boot-starter-aop
72 |
73 |
74 | com.alibaba
75 | druid
76 | 1.1.10
77 |
78 |
79 | org.springframework.boot
80 | spring-boot-starter-thymeleaf
81 |
82 |
83 | org.springframework.boot
84 | spring-boot-starter-json
85 |
86 |
87 | org.springframework.boot
88 | spring-boot-devtools
89 | true
90 |
91 |
92 | mysql
93 | mysql-connector-java
94 |
95 |
96 | org.springframework.boot
97 | spring-boot-configuration-processor
98 | true
99 |
100 |
101 |
102 |
103 |
104 |
105 |
106 | org.apache.maven.plugins
107 | maven-dependency-plugin
108 | 2.8
109 |
110 |
111 | copy-installed
112 | install
113 |
114 | copy
115 |
116 |
117 |
118 |
119 |
120 |
121 | ${project.groupId}
122 | ${project.artifactId}
123 | ${project.version}
124 | ${project.packaging}
125 | true
126 | data-soar.jar
127 |
128 |
129 | dev
130 |
131 |
132 |
133 | org.springframework.boot
134 | spring-boot-maven-plugin
135 |
136 |
137 | org.apache.maven.plugins
138 | maven-surefire-plugin
139 |
140 | false
141 |
142 |
143 |
144 | org.springframework.boot
145 | spring-boot-maven-plugin
146 |
147 |
148 | true
149 |
150 |
151 |
152 |
153 |
154 |
155 |
156 |
--------------------------------------------------------------------------------
/scripts/init.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 | set -eu
3 |
4 | echo 'Check install environment'
5 |
6 | cmd="soar"
7 |
8 | if ! command -v "${cmd}" &> /dev/null ; then
9 | echo -e "\033[91m${cmd} is not set.Eg:\nalias ${cmd}=/root/${cmd}\nexport ${cmd}=/root/${cmd}\033[0m"
10 | wget https://github.com/XiaoMi/soar/releases/download/v0.8.1/soar.linux-amd64 -O soar
11 | chmod a+x soar
12 | go get -d github.com/XiaoMi/soar
13 | cd ${GOPATH}/src/github.com/XiaoMi/soar && make
14 | exit 1
15 | else
16 | echo "${cmd} is ok"
17 | fi
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/DataPlatformSoarApplication.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar;
2 |
3 | import org.springframework.boot.SpringApplication;
4 | import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
5 | import org.springframework.boot.autoconfigure.SpringBootApplication;
6 | import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
7 | import org.springframework.boot.autoconfigure.jdbc.JdbcTemplateAutoConfiguration;
8 |
9 | @SpringBootApplication
10 | @EnableAutoConfiguration(exclude={JdbcTemplateAutoConfiguration.class,DataSourceAutoConfiguration.class})
11 | public class DataPlatformSoarApplication {
12 |
13 | public static void main(String[] args) {
14 | SpringApplication.run(DataPlatformSoarApplication.class, args);
15 | }
16 |
17 | }
18 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/config.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar;
2 |
3 | import org.junit.jupiter.api.Test;
4 | import org.springframework.boot.env.YamlPropertySourceLoader;
5 | import org.springframework.core.env.PropertySource;
6 | import org.springframework.core.io.ClassPathResource;
7 | import org.springframework.core.io.Resource;
8 |
9 | import java.io.IOException;
10 |
11 | public class config {
12 | private final YamlPropertySourceLoader loader = new YamlPropertySourceLoader();
13 |
14 | @Test
15 | public void test () {
16 | Resource path = new ClassPathResource("soar.yml");
17 | PropertySource> propertySource = loadYaml(path);
18 | }
19 |
20 | private PropertySource> loadYaml(Resource path) {
21 | if (!path.exists()) {
22 | throw new IllegalArgumentException("Resource " + path + " does not exist");
23 | }
24 | try {
25 | return this.loader.load("custom-resource", path).get(0);
26 | }
27 | catch (IOException ex) {
28 | throw new IllegalStateException(
29 | "Failed to load yaml configuration from " + path, ex);
30 | }
31 | }
32 | }
33 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/configs/MyWebSocketHandler.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.configs;
2 |
3 | import com.bektz.dataplatformsoar.req.SqlVerifyReq;
4 | import com.bektz.dataplatformsoar.resp.JdbcResultResp;
5 | import com.bektz.dataplatformsoar.resp.SqlVerifyResp;
6 | import com.bektz.dataplatformsoar.service.SchemaService;
7 | import com.bektz.dataplatformsoar.service.SoarService;
8 | import com.bektz.dataplatformsoar.sqlparser.SqlParserService;
9 | import com.fasterxml.jackson.databind.ObjectMapper;
10 | import lombok.extern.slf4j.Slf4j;
11 | import org.apache.commons.lang3.StringUtils;
12 | import org.springframework.beans.factory.annotation.Autowired;
13 | import org.springframework.stereotype.Component;
14 | import org.springframework.web.socket.TextMessage;
15 | import org.springframework.web.socket.WebSocketSession;
16 | import org.springframework.web.socket.handler.TextWebSocketHandler;
17 |
18 | import java.io.IOException;
19 | import java.util.Map;
20 |
21 | import static org.apache.commons.lang3.StringUtils.contains;
22 | import static org.apache.commons.lang3.StringUtils.substringAfter;
23 |
24 | @Slf4j
25 | @Component
26 | public class MyWebSocketHandler extends TextWebSocketHandler {
27 |
28 |
29 | @Autowired
30 | private SoarService soarService;
31 | @Autowired
32 | private SchemaService schemaService;
33 | @Autowired
34 | private SqlParserService sqlParserService;
35 | @Autowired
36 | private ObjectMapper mapper;
37 |
38 | private static final String CONNECTION_HEART_BEAT_MESSAGE = "HeartBeat";
39 | private static final String EXECUTE_QUERY_SQL_MARK = ":::";
40 | private static final String PARSER_SQL_COLUMN_MARK = "===";
41 |
42 | @Override
43 | public void handleTextMessage(WebSocketSession session, TextMessage message) {
44 | if (message.getPayloadLength() == 0 || message.getPayload().equals(CONNECTION_HEART_BEAT_MESSAGE)) {
45 | return;
46 | }
47 | try {
48 | boolean contains = contains(message.getPayload(), EXECUTE_QUERY_SQL_MARK);
49 | //sql执行
50 | if (contains) {
51 | String[] split = StringUtils.split(message.getPayload(), EXECUTE_QUERY_SQL_MARK);
52 | JdbcResultResp jdbcResultResp = schemaService.executeSql(SqlVerifyReq.builder().schema(split[0]).sql(split[1]).build());
53 | sendMessage(session, mapper.writeValueAsString(jdbcResultResp));
54 |
55 | } else if (contains(message.getPayload(), PARSER_SQL_COLUMN_MARK)) {
56 | String s = substringAfter(message.getPayload(), PARSER_SQL_COLUMN_MARK);
57 | Map map = sqlParserService.parserSql(s);
58 | sendMessage(session, mapper.writeValueAsString(map));
59 | } else {
60 | SqlVerifyReq sqlVerifyReq = SqlVerifyReq.builder().sql(message.getPayload()).build();
61 | SqlVerifyResp verify = soarService.verify(sqlVerifyReq);
62 | sendMessage(session, mapper.writeValueAsString(verify));
63 | }
64 |
65 | } catch (Exception e) {
66 | log.error("sql分析或执行sql失败 sql:{}", message.getPayload(), e);
67 | }
68 | }
69 |
70 | private void sendMessage(WebSocketSession webSocketSession, String message) throws IOException {
71 | TextMessage textMessage = new TextMessage(message);
72 | webSocketSession.sendMessage(textMessage);
73 | }
74 |
75 | }
76 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/configs/ParamValidAop.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.configs;
2 |
3 | import com.bektz.dataplatformsoar.resp.GenericResponse;
4 | import lombok.extern.slf4j.Slf4j;
5 | import org.aspectj.lang.ProceedingJoinPoint;
6 | import org.aspectj.lang.annotation.Around;
7 | import org.aspectj.lang.annotation.Aspect;
8 | import org.aspectj.lang.annotation.Pointcut;
9 | import org.springframework.context.annotation.Configuration;
10 | import org.springframework.http.ResponseEntity;
11 | import org.springframework.validation.BindingResult;
12 | import org.springframework.validation.ObjectError;
13 |
14 | import java.util.List;
15 |
16 |
17 | @Slf4j
18 | @Aspect
19 | @Configuration
20 | public class ParamValidAop {
21 |
22 | ThreadLocal startTime = new ThreadLocal<>();
23 |
24 | @Pointcut("within(@org.springframework.web.bind.annotation.RestController *) || within(@org.springframework.stereotype.Controller *)")
25 | public void excude() {
26 | }
27 |
28 | @Around("excude()")
29 | public Object doAround(ProceedingJoinPoint pjp) throws Throwable {
30 | Object[] args = pjp.getArgs();
31 | for (Object arg : args) {
32 | if (arg instanceof BindingResult) {
33 | BindingResult result = (BindingResult) arg;
34 | if (result.getErrorCount() > 0) {
35 | List allErrors = result.getAllErrors();
36 | StringBuilder sb = new StringBuilder();
37 | for (ObjectError allError : allErrors) {
38 | sb.append(allError.getDefaultMessage()).append(";");
39 | }
40 | return ResponseEntity.badRequest().body(GenericResponse.CLIENT_ERROR.setMessage(sb.toString()));
41 | }
42 | }
43 | }
44 | return pjp.proceed();
45 | }
46 |
47 | }
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/configs/SoarConfiguration.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.configs;
2 |
3 | import lombok.Data;
4 | import org.springframework.boot.context.properties.ConfigurationProperties;
5 | import org.springframework.stereotype.Component;
6 |
7 | import java.io.Serializable;
8 | import java.util.Map;
9 |
10 | @Data
11 | @ConfigurationProperties("soar-configs.skills")
12 | @Component
13 | public class SoarConfiguration {
14 |
15 | private Map skillsMap;
16 |
17 | public Map getSkillsMap() {
18 | return skillsMap;
19 | }
20 |
21 | @Data
22 | public static class Skill implements Serializable {
23 | private int type;
24 | private boolean isDefault;
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/configs/WebSocketConfig.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.configs;
2 |
3 | import org.springframework.beans.factory.annotation.Autowired;
4 | import org.springframework.context.annotation.Bean;
5 | import org.springframework.context.annotation.Configuration;
6 | import org.springframework.web.socket.WebSocketHandler;
7 | import org.springframework.web.socket.config.annotation.EnableWebSocket;
8 | import org.springframework.web.socket.config.annotation.WebSocketConfigurer;
9 | import org.springframework.web.socket.config.annotation.WebSocketHandlerRegistry;
10 | import org.springframework.web.socket.server.support.HttpSessionHandshakeInterceptor;
11 |
12 | @Configuration
13 | @EnableWebSocket
14 | public class WebSocketConfig implements WebSocketConfigurer {
15 |
16 | @Autowired
17 | private MyWebSocketHandler myWebSocketHandler;
18 |
19 | @Override
20 | public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
21 | registry.addHandler(myWebSocketHandler, "/myHandler")
22 | .addInterceptors(new HttpSessionHandshakeInterceptor());
23 | }
24 |
25 | @Bean
26 | public WebSocketHandler myHandler() {
27 | return myWebSocketHandler;
28 | }
29 |
30 | }
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/constants/Common.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.constants;
2 |
3 | public final class Common {
4 | public static final String ILLEGAL_INSTANCE_MESSAGE ="非法实例化异常";
5 | }
6 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/constants/JdbcConstants.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.constants;
2 |
3 | public final class JdbcConstants {
4 |
5 | public static final int QUERY_FAILURE_CODE = 5001;
6 | public static final String QUERY_FAILURE_MESSAGE = "查询sql失败,请稍后再试";
7 |
8 | }
9 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/constants/QueryStatus.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.constants;
2 |
3 | import java.util.Objects;
4 |
5 | public enum QueryStatus {
6 |
7 | SUCCESS("查询成功"), FAILURE("查询失败");
8 | public String value;
9 | private QueryStatus(String value) {
10 | this.value = value;
11 | }
12 | public Boolean isSuccess() {
13 | return Objects.equals(SUCCESS, this);
14 | }
15 | }
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/constants/QueryType.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.constants;
2 |
3 | public enum QueryType {
4 | ALL,PART
5 | }
6 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/controller/PageController.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.controller;
2 |
3 |
4 | import com.bektz.dataplatformsoar.resp.SchemaResp;
5 | import com.bektz.dataplatformsoar.service.SchemaService;
6 | import org.springframework.beans.factory.annotation.Autowired;
7 | import org.springframework.stereotype.Controller;
8 | import org.springframework.ui.Model;
9 | import org.springframework.web.bind.annotation.GetMapping;
10 | import org.springframework.web.bind.annotation.PathVariable;
11 |
12 | import java.util.List;
13 |
14 | @Controller
15 | public class PageController {
16 | @Autowired
17 | private SchemaService schemaService;
18 |
19 | @GetMapping(value = {"/index", "/"})
20 | public String index(Model model) {
21 | List schemaResps = schemaService.getSchemaResps();
22 | model.addAttribute("schemaResps", schemaResps);
23 | return "index";
24 | }
25 |
26 |
27 | @GetMapping(value = "/page/{page}")
28 | public String verify(@PathVariable("page") String page) {
29 | return page;
30 | }
31 | }
32 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/controller/SchemaController.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.controller;
2 |
3 | import com.bektz.dataplatformsoar.req.SchemaReq;
4 | import com.bektz.dataplatformsoar.req.SqlVerifyReq;
5 | import com.bektz.dataplatformsoar.resp.GenericResponse;
6 | import com.bektz.dataplatformsoar.resp.JdbcResultResp;
7 | import com.bektz.dataplatformsoar.resp.SchemaResp;
8 | import com.bektz.dataplatformsoar.service.SchemaService;
9 | import org.springframework.beans.factory.annotation.Autowired;
10 | import org.springframework.http.ResponseEntity;
11 | import org.springframework.stereotype.Controller;
12 | import org.springframework.ui.Model;
13 | import org.springframework.validation.BindingResult;
14 | import org.springframework.web.bind.annotation.*;
15 |
16 | import javax.validation.Valid;
17 | import java.util.Collections;
18 | import java.util.List;
19 | import java.util.Map;
20 |
21 |
22 | @Controller
23 | public class SchemaController {
24 |
25 | @Autowired
26 | private SchemaService schemaService;
27 |
28 | @PostMapping(value = "/schemas")
29 | public ResponseEntity addSchema(@RequestBody @Valid SchemaReq schemaReq, BindingResult result) {
30 | schemaService.addSchema(schemaReq);
31 | return ResponseEntity.ok(GenericResponse.SUCCESS);
32 | }
33 |
34 | @GetMapping(value = "/schemas/{schema}")
35 | public String allTablesBySchema(@PathVariable("schema") String shcema, Model model) {
36 | SchemaResp schemaResp = schemaService.getSchemaResp(shcema);
37 | model.addAttribute("schemaResps", Collections.singletonList(schemaResp));
38 | return "index::table_refresh";
39 | }
40 |
41 | @GetMapping(value = "/schemas/all")
42 | public String allSchemas(Model model) {
43 | List schemaResps = schemaService.getSchemaResps();
44 | model.addAttribute("schemaResps", schemaResps);
45 | return "index::schema_refresh";
46 | }
47 |
48 | @PostMapping(value = "/schemas/tables/sql::execute")
49 | public JdbcResultResp executeSql(@RequestBody @Valid SqlVerifyReq sqlVerifyReq, BindingResult result) {
50 | return schemaService.executeSql(sqlVerifyReq);
51 | }
52 |
53 | @ResponseBody
54 | @GetMapping(value = "/schemas/tables/{schema}")
55 | public Map> getablesBySchema(@PathVariable("schema") String schema) {
56 | return schemaService.tablesAndSchemas(schema);
57 | }
58 |
59 | }
60 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/controller/SoarController.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.controller;
2 |
3 | import com.bektz.dataplatformsoar.req.SqlVerifyReq;
4 | import com.bektz.dataplatformsoar.resp.SqlVerifyResp;
5 | import com.bektz.dataplatformsoar.service.SoarService;
6 | import org.springframework.beans.factory.annotation.Autowired;
7 | import org.springframework.stereotype.Controller;
8 | import org.springframework.web.bind.annotation.PostMapping;
9 | import org.springframework.web.bind.annotation.RequestBody;
10 | import org.springframework.web.bind.annotation.ResponseBody;
11 |
12 | @Controller
13 | public class SoarController {
14 | @Autowired
15 | private SoarService soarService;
16 |
17 | @ResponseBody
18 | @PostMapping(value = "/soar/sql::verify")
19 | public SqlVerifyResp verify(@RequestBody SqlVerifyReq sqlVerifyReq) {
20 | return soarService.verify(sqlVerifyReq);
21 | }
22 |
23 | }
24 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/controller/handler/ControllerExceptionHandleAdvice.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.controller.handler;
2 |
3 | import com.bektz.dataplatformsoar.exception.BektzClientException;
4 | import com.bektz.dataplatformsoar.exception.BektzException;
5 | import com.bektz.dataplatformsoar.exception.BektzServerException;
6 | import com.bektz.dataplatformsoar.resp.GenericResponse;
7 | import lombok.extern.slf4j.Slf4j;
8 | import org.springframework.http.HttpStatus;
9 | import org.springframework.http.ResponseEntity;
10 | import org.springframework.stereotype.Controller;
11 | import org.springframework.web.bind.annotation.ControllerAdvice;
12 | import org.springframework.web.bind.annotation.ExceptionHandler;
13 | import org.springframework.web.bind.annotation.RestController;
14 |
15 | @Slf4j
16 | @ControllerAdvice(annotations = {RestController.class, Controller.class})
17 | public class ControllerExceptionHandleAdvice {
18 |
19 | @ExceptionHandler
20 | public ResponseEntity handleException(BektzException e) {
21 | GenericResponse genericResponse = GenericResponse.builder().code(e.getCode()).status(e.getStatus()).message(e.getMessage()).build();
22 | if (e instanceof BektzServerException) {
23 | log.error("[ServerError]: ", e);
24 | } else if (e instanceof BektzClientException) {
25 | log.warn("[ClientWarn]: ", e);
26 | } else {
27 | return new ResponseEntity<>(GenericResponse.SERVER_ERROR, HttpStatus.INTERNAL_SERVER_ERROR);
28 | }
29 | return new ResponseEntity<>(genericResponse, HttpStatus.valueOf(e.getStatus()));
30 | }
31 |
32 | @ExceptionHandler
33 | public ResponseEntity handleException(Exception e) {
34 | GenericResponse genericResponse = GenericResponse.builder().code(500).status(500).message("系统繁忙!请稍后再试").build();
35 | return new ResponseEntity<>(genericResponse, HttpStatus.valueOf(500));
36 | }
37 |
38 | }
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/exception/BektzClientException.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.exception;
2 |
3 | import org.springframework.http.HttpStatus;
4 |
5 | public class BektzClientException extends BektzException {
6 |
7 |
8 | public BektzClientException(int code, String message) {
9 | super(HttpStatus.UNPROCESSABLE_ENTITY.value(), code, message);
10 | }
11 |
12 | public BektzClientException(int status, int code, String message) {
13 | super(status, code, message);
14 | }
15 |
16 | public BektzClientException(String message, Throwable cause) {
17 | super(message, cause);
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/exception/BektzException.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.exception;
2 |
3 | import lombok.Data;
4 |
5 | @Data
6 | public class BektzException extends RuntimeException {
7 | private int code;
8 | private int status;
9 | private String message;
10 |
11 | public BektzException(String message, Throwable throwable) {
12 | super(message, throwable);
13 | }
14 |
15 | public BektzException(int code, String message) {
16 | this.code = code;
17 | this.message = message;
18 | }
19 |
20 | public BektzException(int status, int code, String message) {
21 | this.status = status;
22 | this.code = code;
23 | this.message = message;
24 | }
25 |
26 | public BektzException(int code, String message, Throwable cause) {
27 | super(message, cause);
28 | this.code = code;
29 | this.message = message;
30 | }
31 |
32 | }
33 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/exception/BektzServerException.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.exception;
2 |
3 | import org.springframework.http.HttpStatus;
4 |
5 | public class BektzServerException extends BektzException {
6 |
7 |
8 | public BektzServerException(int code, String message) {
9 | super(HttpStatus.INTERNAL_SERVER_ERROR.value(), code, message);
10 | }
11 |
12 | public BektzServerException(int status, int code, String message) {
13 | super(status, code, message);
14 | }
15 |
16 | public BektzServerException(int code, String message, Throwable cause) {
17 | super(message, cause);
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/exception/SqlParserException.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.exception;
2 |
3 | import java.io.PrintStream;
4 | import java.io.PrintWriter;
5 |
6 | public class SqlParserException extends BektzClientException {
7 |
8 | private static final long serialVersionUID = -1099039459759769980L;
9 |
10 | private Throwable cause = null;
11 |
12 |
13 | public SqlParserException(String arg0) {
14 | super(2001, arg0);
15 | }
16 |
17 |
18 | public SqlParserException(String arg0, Throwable arg1) {
19 | super(arg0, arg1);
20 | this.cause = arg1;
21 | }
22 |
23 | @Override
24 | public Throwable getCause() {
25 | return this.cause;
26 | }
27 |
28 | @Override
29 | public void printStackTrace() {
30 | this.printStackTrace(System.err);
31 | }
32 |
33 | @Override
34 | public void printStackTrace(PrintWriter pw) {
35 | super.printStackTrace(pw);
36 | if (this.cause != null) {
37 | pw.println("Caused by:");
38 | this.cause.printStackTrace(pw);
39 | }
40 |
41 | }
42 |
43 | @Override
44 | public void printStackTrace(PrintStream ps) {
45 | super.printStackTrace(ps);
46 | if (this.cause != null) {
47 | ps.println("Caused by:");
48 | this.cause.printStackTrace(ps);
49 | }
50 |
51 | }
52 |
53 |
54 | }
55 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/repository/SecretColumnRepository.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.repository;
2 |
3 | import com.bektz.dataplatformsoar.repository.domain.SecretColumn;
4 | import org.springframework.data.jpa.repository.JpaRepository;
5 |
6 | import java.util.List;
7 |
8 | public interface SecretColumnRepository extends JpaRepository {
9 |
10 | public SecretColumn getSecretColumnByColumnAndTableAndSchemaName(String column, String table, String schemaName);
11 |
12 | public List getSecretColumnsByTableAndSchemaName(String table, String schemaName);
13 | }
14 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/repository/domain/SecretColumn.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.repository.domain;
2 |
3 | import lombok.AllArgsConstructor;
4 | import lombok.Builder;
5 | import lombok.Data;
6 | import lombok.NoArgsConstructor;
7 |
8 | @Builder
9 | @Data
10 | @AllArgsConstructor
11 | @NoArgsConstructor
12 | public class SecretColumn {
13 |
14 | private Long id;
15 |
16 | //数据库名
17 | private String schemaName;
18 |
19 | //表名
20 | private String table;
21 |
22 | //字段
23 | private String column;
24 |
25 | //是否敏感字段:1-敏感,0-非敏感
26 | private Boolean isSecret;
27 |
28 | //字段类型(MOBILE, CARD_NO, ID_NO)
29 | private String columnType;
30 |
31 | //敏感类型:SECRECT(敏感), SUSPECTED_SECRET(疑似敏感)
32 | private String secretType;
33 |
34 | private String createdBy;
35 |
36 | private String updatedBy;
37 | }
38 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/req/SchemaReq.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.req;
2 |
3 | import lombok.AllArgsConstructor;
4 | import lombok.Builder;
5 | import lombok.Data;
6 | import lombok.NoArgsConstructor;
7 |
8 | import javax.validation.constraints.NotBlank;
9 | import javax.validation.constraints.NotNull;
10 |
11 | @Builder
12 | @Data
13 | @AllArgsConstructor
14 | @NoArgsConstructor
15 | public class SchemaReq {
16 |
17 | @NotBlank(message = "数据库地址不能为空")
18 | private String address;
19 | @NotBlank(message = "数据库不能为空")
20 | private String schema;
21 | @NotBlank(message = "用户名不能为空")
22 | private String username;
23 | @NotNull(message = "密码不能为空")
24 | private String password;
25 | @NotNull(message = "端口不能为空")
26 | private Integer port;
27 | }
28 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/req/SqlVerifyReq.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.req;
2 |
3 | import lombok.*;
4 |
5 | import javax.validation.constraints.NotBlank;
6 |
7 | @Getter
8 | @NoArgsConstructor
9 | @AllArgsConstructor
10 | @Setter
11 | @Builder
12 | public class SqlVerifyReq {
13 | @NotBlank(message = "sql语句不能为空")
14 | private String sql;
15 | @NotBlank(message = "请选择数据库后再执行sql")
16 | private String schema;
17 | }
18 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/resp/GenericResponse.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.resp;
2 |
3 | import lombok.AllArgsConstructor;
4 | import lombok.Builder;
5 | import lombok.Data;
6 | import lombok.NoArgsConstructor;
7 |
8 | import static org.apache.logging.log4j.util.Strings.EMPTY;
9 |
10 | @Data
11 | @Builder
12 | @NoArgsConstructor
13 | @AllArgsConstructor
14 | public class GenericResponse {
15 | private int code = 0;
16 |
17 | private String message = EMPTY;
18 |
19 | @Builder.Default
20 | private Boolean succeed = false;
21 |
22 | private int status = 200;
23 |
24 | public static final GenericResponse SUCCESS = new GenericResponse(0, "处理成功", true, 200);
25 |
26 | public static final GenericResponse SERVER_ERROR = new GenericResponse(1001, "服务器异常,请稍后再试", false, 500);
27 |
28 | public static final GenericResponse CLIENT_ERROR = new GenericResponse(2001, "参数不正确", false, 400);
29 |
30 | public GenericResponse setMessage(String message) {
31 | this.message = message;
32 | return this;
33 | }
34 | }
35 |
--------------------------------------------------------------------------------
/src/main/java/com/bektz/dataplatformsoar/resp/JdbcResultResp.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.resp;
2 |
3 | import com.bektz.dataplatformsoar.constants.QueryStatus;
4 | import lombok.AllArgsConstructor;
5 | import lombok.Builder;
6 | import lombok.Data;
7 | import lombok.NoArgsConstructor;
8 | import lombok.experimental.Accessors;
9 |
10 | import java.util.ArrayList;
11 | import java.util.List;
12 | import java.util.Map;
13 |
14 | @Data
15 | @Builder
16 | @AllArgsConstructor
17 | @NoArgsConstructor
18 | @Accessors(chain = true)
19 | public class JdbcResultResp {
20 | @Builder.Default
21 | private List resultMeta = new ArrayList<>();
22 | @Builder.Default
23 | private List
'):x.find("."+r).remove()},"keyup"),""===t&&x.find("."+r).remove(),void $())};f&&k.on("keyup",j).on("blur",function(i){var a=p[0].selectedIndex;e=k,d=t(p[0].options[a]).html(),0===a&&d===k.attr("placeholder")&&(d=""),setTimeout(function(){T(k.val(),function(e){d||k.val("")},"blur")},200)}),g.on("click",function(){var e=t(this),a=e.attr("lay-value"),n=p.attr("lay-filter");return!e.hasClass(c)&&(e.hasClass("layui-select-tips")?k.val(""):(k.val(e.text()),e.addClass(s)),e.siblings().removeClass(s),p.val(a).removeClass("layui-form-danger"),layui.event.call(this,l,"select("+n+")",{elem:p[0],value:a,othis:i}),w(!0),!1)}),i.find("dl>dt").on("click",function(e){return!1}),t(document).off("click",v).on("click",v)}};f.each(function(e,l){var r=t(this),o=r.next("."+a),u=this.disabled,d=l.value,f=t(l.options[l.selectedIndex]),v=l.options[0];if("string"==typeof r.attr("lay-ignore"))return r.show();var h="string"==typeof r.attr("lay-search"),p=v?v.value?i:v.innerHTML||i:i,m=t(['','
','','
','
',function(e){var t=[];return layui.each(e,function(e,a){0!==e||a.value?"optgroup"===a.tagName.toLowerCase()?t.push("- "+a.label+"
"):t.push('- '+a.innerHTML+"
"):t.push('- '+(a.innerHTML||i)+"
")}),0===t.length&&t.push('- 没有选项
'),t.join("")}(r.find("*"))+"
","
"].join(""));o[0]&&o.remove(),r.after(m),y.call(this,m,u,h)})},checkbox:function(){var e={checkbox:["layui-form-checkbox","layui-form-checked","checkbox"],_switch:["layui-form-switch","layui-form-onswitch","switch"]},i=u.find("input[type=checkbox]"),a=function(e,i){var a=t(this);e.on("click",function(){var t=a.attr("lay-filter"),n=(a.attr("lay-text")||"").split("|");a[0].disabled||(a[0].checked?(a[0].checked=!1,e.removeClass(i[1]).find("em").text(n[1])):(a[0].checked=!0,e.addClass(i[1]).find("em").text(n[0])),layui.event.call(a[0],l,i[2]+"("+t+")",{elem:a[0],value:a[0].value,othis:e}))})};i.each(function(i,n){var l=t(this),r=l.attr("lay-skin"),s=(l.attr("lay-text")||"").split("|"),o=this.disabled;"switch"===r&&(r="_"+r);var u=e[r]||e.checkbox;if("string"==typeof l.attr("lay-ignore"))return l.show();var d=l.next("."+u[0]),f=t(['",function(){var e=n.title.replace(/\s/g,""),t={checkbox:[e?""+n.title+"":"",''].join(""),_switch:""+((n.checked?s[0]:s[1])||"")+""};return t[r]||t.checkbox}(),"
"].join(""));d[0]&&d.remove(),l.after(f),a.call(this,f,u)})},radio:function(){var e="layui-form-radio",i=["",""],a=u.find("input[type=radio]"),n=function(a){var n=t(this),s="layui-anim-scaleSpring";a.on("click",function(){var o=n[0].name,c=n.parents(r),u=n.attr("lay-filter"),d=c.find("input[name="+o.replace(/(\.|#|\[|\])/g,"\\$1")+"]");n[0].disabled||(layui.each(d,function(){var a=t(this).next("."+e);this.checked=!1,a.removeClass(e+"ed"),a.find(".layui-icon").removeClass(s).html(i[1])}),n[0].checked=!0,a.addClass(e+"ed"),a.find(".layui-icon").addClass(s).html(i[0]),layui.event.call(n[0],l,"radio("+u+")",{elem:n[0],value:n[0].value,othis:a}))})};a.each(function(a,l){var r=t(this),s=r.next("."+e),o=this.disabled;if("string"==typeof r.attr("lay-ignore"))return r.show();s[0]&&s.remove();var u=t(['','
'+i[l.checked?0:1]+"","
"+function(){var e=l.title||"";return"string"==typeof r.next().attr("lay-radio")&&(e=r.next().html(),r.next().remove()),e}()+"
","
"].join(""));r.after(u),n.call(this,u)})}};return e?d[e]?d[e]():a.error("不支持的"+e+"表单渲染"):layui.each(d,function(e,t){t()}),n};var d=function(){var e=t(this),a=f.config.verify,s=null,o="layui-form-danger",c={},u=e.parents(r),d=u.find("*[lay-verify]"),v=e.parents("form")[0],h=u.find("input,select,textarea"),y=e.attr("lay-filter");if(layui.each(d,function(e,l){var r=t(this),c=r.attr("lay-verify").split("|"),u=r.attr("lay-verType"),d=r.val();if(r.removeClass(o),layui.each(c,function(e,t){var c,f="",v="function"==typeof a[t];if(a[t]){var c=v?f=a[t](d,l):!a[t][0].test(d);if(f=f||a[t][1],c)return"tips"===u?i.tips(f,function(){return"string"==typeof r.attr("lay-ignore")||"select"!==l.tagName.toLowerCase()&&!/^checkbox|radio$/.test(l.type)?r:r.next()}(),{tips:1}):"alert"===u?i.alert(f,{title:"提示",shadeClose:!0}):i.msg(f,{icon:5,shift:6}),n.android||n.ios||l.focus(),r.addClass(o),s=!0}}),s)return s}),s)return!1;var p={};return layui.each(h,function(e,t){if(t.name=(t.name||"").replace(/^\s*|\s*&/,""),t.name){if(/^.*\[\]$/.test(t.name)){var i=t.name.match(/^(.*)\[\]$/g)[0];p[i]=0|p[i],t.name=t.name.replace(/^(.*)\[\]$/,"$1["+p[i]++ +"]")}/^checkbox|radio$/.test(t.type)&&!t.checked||(c[t.name]=t.value)}}),layui.event.call(this,l,"submit("+y+")",{elem:this,form:v,field:c})},f=new u,v=t(document),h=t(window);f.render(),v.on("reset",r,function(){var e=t(this).attr("lay-filter");setTimeout(function(){f.render(null,e)},50)}),v.on("submit",r,d).on("click","*[lay-submit]",d),e(l,f)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/laypage.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define(function(e){"use strict";var a=document,t="getElementById",n="getElementsByTagName",i="laypage",r="layui-disabled",u=function(e){var a=this;a.config=e||{},a.config.index=++s.index,a.render(!0)};u.prototype.type=function(){var e=this.config;if("object"==typeof e.elem)return void 0===e.elem.length?2:3},u.prototype.view=function(){var e=this,a=e.config,t=a.groups="groups"in a?0|a.groups:5;a.layout="object"==typeof a.layout?a.layout:["prev","page","next"],a.count=0|a.count,a.curr=0|a.curr||1,a.limits="object"==typeof a.limits?a.limits:[10,20,30,40,50],a.limit=0|a.limit||10,a.pages=Math.ceil(a.count/a.limit)||1,a.curr>a.pages&&(a.curr=a.pages),t<0?t=1:t>a.pages&&(t=a.pages),a.prev="prev"in a?a.prev:"上一页",a.next="next"in a?a.next:"下一页";var n=a.pages>t?Math.ceil((a.curr+(t>1?1:0))/(t>0?t:1)):1,i={prev:function(){return a.prev?''+a.prev+"":""}(),page:function(){var e=[];if(a.count<1)return"";n>1&&a.first!==!1&&0!==t&&e.push(''+(a.first||1)+"");var i=Math.floor((t-1)/2),r=n>1?a.curr-i:1,u=n>1?function(){var e=a.curr+(t-i-1);return e>a.pages?a.pages:e}():t;for(u-r2&&e.push('…');r<=u;r++)r===a.curr?e.push('"+r+""):e.push(''+r+"");return a.pages>t&&a.pages>u&&a.last!==!1&&(u+1…'),0!==t&&e.push(''+(a.last||a.pages)+"")),e.join("")}(),next:function(){return a.next?''+a.next+"":""}(),count:'共 '+a.count+" 条",limit:function(){var e=['"}(),refresh:['','',""].join(""),skip:function(){return['到第','','页',""].join("")}()};return['',function(){var e=[];return layui.each(a.layout,function(a,t){i[t]&&e.push(i[t])}),e.join("")}(),"
"].join("")},u.prototype.jump=function(e,a){if(e){var t=this,i=t.config,r=e.children,u=e[n]("button")[0],l=e[n]("input")[0],p=e[n]("select")[0],c=function(){var e=0|l.value.replace(/\s|\D/g,"");e&&(i.curr=e,t.render())};if(a)return c();for(var o=0,y=r.length;oi.pages||(i.curr=e,t.render())});p&&s.on(p,"change",function(){var e=this.value;i.curr*e>i.count&&(i.curr=Math.ceil(i.count/e)),i.limit=e,t.render()}),u&&s.on(u,"click",function(){c()})}},u.prototype.skip=function(e){if(e){var a=this,t=e[n]("input")[0];t&&s.on(t,"keyup",function(t){var n=this.value,i=t.keyCode;/^(37|38|39|40)$/.test(i)||(/\D/.test(n)&&(this.value=n.replace(/\D/,"")),13===i&&a.jump(e,!0))})}},u.prototype.render=function(e){var n=this,i=n.config,r=n.type(),u=n.view();2===r?i.elem&&(i.elem.innerHTML=u):3===r?i.elem.html(u):a[t](i.elem)&&(a[t](i.elem).innerHTML=u),i.jump&&i.jump(i,e);var s=a[t]("layui-laypage-"+i.index);n.jump(s),i.hash&&!e&&(location.hash="!"+i.hash+"="+i.curr),n.skip(s)};var s={render:function(e){var a=new u(e);return a.index},index:layui.laypage?layui.laypage.index+1e4:0,on:function(e,a,t){return e.attachEvent?e.attachEvent("on"+a,function(a){a.target=a.srcElement,t.call(e,a)}):e.addEventListener(a,t,!1),this}};e(i,s)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/laytpl.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define(function(e){"use strict";var r={open:"{{",close:"}}"},c={exp:function(e){return new RegExp(e,"g")},query:function(e,c,t){var o=["#([\\s\\S])+?","([^{#}])*?"][e||0];return n((c||"")+r.open+o+r.close+(t||""))},escape:function(e){return String(e||"").replace(/&(?!#?[a-zA-Z0-9]+;)/g,"&").replace(//g,">").replace(/'/g,"'").replace(/"/g,""")},error:function(e,r){var c="Laytpl Error:";return"object"==typeof console&&console.error(c+e+"\n"+(r||"")),c+e}},n=c.exp,t=function(e){this.tpl=e};t.pt=t.prototype,window.errors=0,t.pt.parse=function(e,t){var o=this,p=e,a=n("^"+r.open+"#",""),l=n(r.close+"$","");e=e.replace(/\s+|\r|\t|\n/g," ").replace(n(r.open+"#"),r.open+"# ").replace(n(r.close+"}"),"} "+r.close).replace(/\\/g,"\\\\").replace(n(r.open+"!(.+?)!"+r.close),function(e){return e=e.replace(n("^"+r.open+"!"),"").replace(n("!"+r.close),"").replace(n(r.open+"|"+r.close),function(e){return e.replace(/(.)/g,"\\$1")})}).replace(/(?="|')/g,"\\").replace(c.query(),function(e){return e=e.replace(a,"").replace(l,""),'";'+e.replace(/\\/g,"")+';view+="'}).replace(c.query(1),function(e){var c='"+(';return e.replace(/\s/g,"")===r.open+r.close?"":(e=e.replace(n(r.open+"|"+r.close),""),/^=/.test(e)&&(e=e.replace(/^=/,""),c='"+_escape_('),c+e.replace(/\\/g,"")+')+"')}),e='"use strict";var view = "'+e+'";return view;';try{return o.cache=e=new Function("d, _escape_",e),e(t,c.escape)}catch(u){return delete o.cache,c.error(u,p)}},t.pt.render=function(e,r){var n,t=this;return e?(n=t.cache?t.cache(e,c.escape):t.parse(t.tpl,e),r?void r(n):n):c.error("no data")};var o=function(e){return"string"!=typeof e?c.error("Template not found"):new t(e)};o.config=function(e){e=e||{};for(var c in e)r[c]=e[c]},o.v="1.2.0",e("laytpl",o)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/rate.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define("jquery",function(e){"use strict";var a=layui.jquery,i={config:{},index:layui.rate?layui.rate.index+1e4:0,set:function(e){var i=this;return i.config=a.extend({},i.config,e),i},on:function(e,a){return layui.onevent.call(this,n,e,a)}},l=function(){var e=this,a=e.config;return{setvalue:function(a){e.setvalue.call(e,a)},config:a}},n="rate",t="layui-rate",o="layui-icon-rate",s="layui-icon-rate-solid",u="layui-icon-rate-half",r="layui-icon-rate-solid layui-icon-rate-half",c="layui-icon-rate-solid layui-icon-rate",f="layui-icon-rate layui-icon-rate-half",v=function(e){var l=this;l.index=++i.index,l.config=a.extend({},l.config,i.config,e),l.render()};v.prototype.config={length:5,text:!1,readonly:!1,half:!1,value:0,theme:""},v.prototype.render=function(){var e=this,i=e.config,l=i.theme?'style="color: '+i.theme+';"':"";i.elem=a(i.elem),parseInt(i.value)!==i.value&&(i.half||(i.value=Math.ceil(i.value)-i.value<.5?Math.ceil(i.value):Math.floor(i.value)));for(var n='",u=1;u<=i.length;u++){var r=' ";i.half&&parseInt(i.value)!==i.value&&u==Math.ceil(i.value)?n=n+' ":n+=r}n+="
"+(i.text?''+i.value+"星":"")+"";var c=i.elem,f=c.next("."+t);f[0]&&f.remove(),e.elemTemp=a(n),i.span=e.elemTemp.next("span"),i.setText&&i.setText(i.value),c.html(e.elemTemp),c.addClass("layui-inline"),i.readonly||e.action()},v.prototype.setvalue=function(e){var a=this,i=a.config;i.value=e,a.render()},v.prototype.action=function(){var e=this,i=e.config,l=e.elemTemp,n=l.find("i").width();l.children("li").each(function(e){var t=e+1,v=a(this);v.on("click",function(e){if(i.value=t,i.half){var o=e.pageX-a(this).offset().left;o<=n/2&&(i.value=i.value-.5)}i.text&&l.next("span").text(i.value+"星"),i.choose&&i.choose(i.value),i.setText&&i.setText(i.value)}),v.on("mousemove",function(e){if(l.find("i").each(function(){a(this).addClass(o).removeClass(r)}),l.find("i:lt("+t+")").each(function(){a(this).addClass(s).removeClass(f)}),i.half){var c=e.pageX-a(this).offset().left;c<=n/2&&v.children("i").addClass(u).removeClass(s)}}),v.on("mouseleave",function(){l.find("i").each(function(){a(this).addClass(o).removeClass(r)}),l.find("i:lt("+Math.floor(i.value)+")").each(function(){a(this).addClass(s).removeClass(f)}),i.half&&parseInt(i.value)!==i.value&&l.children("li:eq("+Math.floor(i.value)+")").children("i").addClass(u).removeClass(c)})})},v.prototype.events=function(){var e=this;e.config},i.render=function(e){var a=new v(e);return l.call(a)},e(n,i)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/slider.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define("jquery",function(e){"use strict";var i=layui.jquery,t={config:{},index:layui.slider?layui.slider.index+1e4:0,set:function(e){var t=this;return t.config=i.extend({},t.config,e),t},on:function(e,i){return layui.onevent.call(this,n,e,i)}},a=function(){var e=this,i=e.config;return{setValue:function(i,t){return e.slide("set",i,t||0)},config:i}},n="slider",l="layui-disabled",s="layui-slider",r="layui-slider-bar",o="layui-slider-wrap",u="layui-slider-wrap-btn",d="layui-slider-tips",v="layui-slider-input",c="layui-slider-input-txt",m="layui-slider-input-btn",p="layui-slider-hover",f=function(e){var a=this;a.index=++t.index,a.config=i.extend({},a.config,t.config,e),a.render()};f.prototype.config={type:"default",min:0,max:100,value:0,step:1,showstep:!1,tips:!0,input:!1,range:!1,height:200,disabled:!1,theme:"#009688"},f.prototype.render=function(){var e=this,t=e.config;if(t.step<1&&(t.step=1),t.maxt.min?a:t.min,t.value[1]=n>t.min?n:t.min,t.value[0]=t.value[0]>t.max?t.max:t.value[0],t.value[1]=t.value[1]>t.max?t.max:t.value[1];var r=Math.floor((t.value[0]-t.min)/(t.max-t.min)*100),v=Math.floor((t.value[1]-t.min)/(t.max-t.min)*100),m=v-r+"%";r+="%",v+="%"}else{"object"==typeof t.value&&(t.value=Math.min.apply(null,t.value)),t.valuet.max&&(t.value=t.max);var m=Math.floor((t.value-t.min)/(t.max-t.min)*100)+"%"}var p=t.disabled?"#c2c2c2":t.theme,f=''+(t.tips?'
':"")+'
'+(t.range?'
':"")+"
",h=i(t.elem),y=h.next("."+s);if(y[0]&&y.remove(),e.elemTemp=i(f),t.range?(e.elemTemp.find("."+o).eq(0).data("value",t.value[0]),e.elemTemp.find("."+o).eq(1).data("value",t.value[1])):e.elemTemp.find("."+o).data("value",t.value),h.html(e.elemTemp),"vertical"===t.type&&e.elemTemp.height(t.height+"px"),t.showstep){for(var g=(t.max-t.min)/t.step,b="",x=1;x')}e.elemTemp.append(b)}if(t.input&&!t.range){var w=i('');h.css("position","relative"),h.append(w),h.find("."+c).children("input").val(t.value),"vertical"===t.type?w.css({left:0,top:-48}):e.elemTemp.css("margin-right",w.outerWidth()+15)}t.disabled?(e.elemTemp.addClass(l),e.elemTemp.find("."+u).addClass(l)):e.slide(),e.elemTemp.find("."+u).on("mouseover",function(){var a="vertical"===t.type?t.height:e.elemTemp[0].offsetWidth,n=e.elemTemp.find("."+o),l="vertical"===t.type?a-i(this).parent()[0].offsetTop-n.height():i(this).parent()[0].offsetLeft,s=l/a*100,r=i(this).parent().data("value"),u=t.setTips?t.setTips(r):r;e.elemTemp.find("."+d).html(u),"vertical"===t.type?e.elemTemp.find("."+d).css({bottom:s+"%","margin-bottom":"20px",display:"inline-block"}):e.elemTemp.find("."+d).css({left:s+"%",display:"inline-block"})}).on("mouseout",function(){e.elemTemp.find("."+d).css("display","none")})},f.prototype.slide=function(e,t,a){var n=this,l=n.config,s=n.elemTemp,f=function(){return"vertical"===l.type?l.height:s[0].offsetWidth},h=s.find("."+o),y=s.next("."+v),g=y.children("."+c).children("input").val(),b=100/((l.max-l.min)/Math.ceil(l.step)),x=function(e,i){e=Math.ceil(e)*b>100?Math.ceil(e)*b:Math.round(e)*b,e=e>100?100:e,h.eq(i).css("vertical"===l.type?"bottom":"left",e+"%");var t=T(h[0].offsetLeft),a=l.range?T(h[1].offsetLeft):0;"vertical"===l.type?(s.find("."+d).css({bottom:e+"%","margin-bottom":"20px"}),t=T(f()-h[0].offsetTop-h.height()),a=l.range?T(f()-h[1].offsetTop-h.height()):0):s.find("."+d).css("left",e+"%"),t=t>100?100:t,a=a>100?100:a;var n=Math.min(t,a),o=Math.abs(t-a);"vertical"===l.type?s.find("."+r).css({height:o+"%",bottom:n+"%"}):s.find("."+r).css({width:o+"%",left:n+"%"});var u=l.min+Math.round((l.max-l.min)*e/100);if(g=u,y.children("."+c).children("input").val(g),h.eq(i).data("value",u),u=l.setTips?l.setTips(u):u,s.find("."+d).html(u),l.range){var v=[h.eq(0).data("value"),h.eq(1).data("value")];v[0]>v[1]&&v.reverse()}l.change&&l.change(l.range?v:u)},T=function(e){var i=e/f()*100/b,t=Math.round(i)*b;return e==f()&&(t=Math.ceil(i)*b),t},w=i(['f()&&(r=f());var o=r/f()*100/b;x(o,e),t.addClass(p),s.find("."+d).show(),i.preventDefault()},o=function(){t.removeClass(p),s.find("."+d).hide()};M(r,o)})}),s.on("click",function(e){var t=i("."+u);if(!t.is(event.target)&&0===t.has(event.target).length&&t.length){var a,n="vertical"===l.type?f()-e.clientY+i(this).offset().top:e.clientX-i(this).offset().left;n<0&&(n=0),n>f()&&(n=f());var s=n/f()*100/b;a=l.range?"vertical"===l.type?Math.abs(n-parseInt(i(h[0]).css("bottom")))>Math.abs(n-parseInt(i(h[1]).css("bottom")))?1:0:Math.abs(n-h[0].offsetLeft)>Math.abs(n-h[1].offsetLeft)?1:0:0,x(s,a),e.preventDefault()}}),y.hover(function(){var e=i(this);e.children("."+m).fadeIn("fast")},function(){var e=i(this);e.children("."+m).fadeOut("fast")}),y.children("."+m).children("i").each(function(e){i(this).on("click",function(){g=1==e?g-l.stepl.max?l.max:Number(g)+l.step;var i=(g-l.min)/(l.max-l.min)*100/b;x(i,0)})});var q=function(){var e=this.value;e=isNaN(e)?0:e,e=el.max?l.max:e,this.value=e;var i=(e-l.min)/(l.max-l.min)*100/b;x(i,0)};y.children("."+c).children("input").on("keydown",function(e){13===e.keyCode&&(e.preventDefault(),q.call(this))}).on("change",q)},f.prototype.events=function(){var e=this;e.config},t.render=function(e){var i=new f(e);return a.call(i)},e(n,t)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/tree.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define("jquery",function(e){"use strict";var o=layui.$,a=layui.hint(),i="layui-tree-enter",r=function(e){this.options=e},t={arrow:["",""],checkbox:["",""],radio:["",""],branch:["",""],leaf:""};r.prototype.init=function(e){var o=this;e.addClass("layui-box layui-tree"),o.options.skin&&e.addClass("layui-tree-skin-"+o.options.skin),o.tree(e),o.on(e)},r.prototype.tree=function(e,a){var i=this,r=i.options,n=a||r.nodes;layui.each(n,function(a,n){var l=n.children&&n.children.length>0,c=o(''),s=o(["",function(){return l?''+(n.spread?t.arrow[1]:t.arrow[0])+"":""}(),function(){return r.check?''+("checkbox"===r.check?t.checkbox[0]:"radio"===r.check?t.radio[0]:"")+"":""}(),function(){return'"+(''+(l?n.spread?t.branch[1]:t.branch[0]:t.leaf)+"")+(""+(n.name||"未命名")+"")}(),""].join(""));l&&(s.append(c),i.tree(c,n.children)),e.append(s),"function"==typeof r.click&&i.click(s,n),i.spread(s,n),r.drag&&i.drag(s,n)})},r.prototype.click=function(e,o){var a=this,i=a.options;e.children("a").on("click",function(e){layui.stope(e),i.click(o)})},r.prototype.spread=function(e,o){var a=this,i=(a.options,e.children(".layui-tree-spread")),r=e.children("ul"),n=e.children("a"),l=function(){e.data("spread")?(e.data("spread",null),r.removeClass("layui-show"),i.html(t.arrow[0]),n.find(".layui-icon").html(t.branch[0])):(e.data("spread",!0),r.addClass("layui-show"),i.html(t.arrow[1]),n.find(".layui-icon").html(t.branch[1]))};r[0]&&(i.on("click",l),n.on("dblclick",l))},r.prototype.on=function(e){var a=this,r=a.options,t="layui-tree-drag";e.find("i").on("selectstart",function(e){return!1}),r.drag&&o(document).on("mousemove",function(e){var i=a.move;if(i.from){var r=(i.to,o(''));e.preventDefault(),o("."+t)[0]||o("body").append(r);var n=o("."+t)[0]?o("."+t):r;n.addClass("layui-show").html(i.from.elem.children("a").html()),n.css({left:e.pageX+10,top:e.pageY+10})}}).on("mouseup",function(){var e=a.move;e.from&&(e.from.elem.children("a").removeClass(i),e.to&&e.to.elem.children("a").removeClass(i),a.move={},o("."+t).remove())})},r.prototype.move={},r.prototype.drag=function(e,a){var r=this,t=(r.options,e.children("a")),n=function(){var t=o(this),n=r.move;n.from&&(n.to={item:a,elem:e},t.addClass(i))};t.on("mousedown",function(){var o=r.move;o.from={item:a,elem:e}}),t.on("mouseenter",n).on("mousemove",n).on("mouseleave",function(){var e=o(this),a=r.move;a.from&&(delete a.to,e.removeClass(i))})},e("tree",function(e){var i=new r(e=e||{}),t=o(e.elem);return t[0]?void i.init(t):a.error("layui.tree 没有找到"+e.elem+"元素")})});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/upload.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define("layer",function(e){"use strict";var i=layui.$,t=layui.layer,n=layui.hint(),a=layui.device(),o={config:{},set:function(e){var t=this;return t.config=i.extend({},t.config,e),t},on:function(e,i){return layui.onevent.call(this,r,e,i)}},l=function(){var e=this;return{upload:function(i){e.upload.call(e,i)},config:e.config}},r="upload",u="layui-upload-file",c="layui-upload-form",f="layui-upload-iframe",s="layui-upload-choose",p=function(e){var t=this;t.config=i.extend({},t.config,o.config,e),t.render()};p.prototype.config={accept:"images",exts:"",auto:!0,bindAction:"",url:"",field:"file",method:"post",data:{},drag:!0,size:0,number:0,multiple:!1},p.prototype.render=function(e){var t=this,e=t.config;e.elem=i(e.elem),e.bindAction=i(e.bindAction),t.file(),t.events()},p.prototype.file=function(){var e=this,t=e.config,n=e.elemFile=i(['"].join("")),o=t.elem.next();(o.hasClass(u)||o.hasClass(c))&&o.remove(),a.ie&&a.ie<10&&t.elem.wrap(''),e.isFile()?(e.elemFile=t.elem,t.field=t.elem[0].name):t.elem.after(n),a.ie&&a.ie<10&&e.initIE()},p.prototype.initIE=function(){var e=this,t=e.config,n=i(''),a=i(['"].join(""));i("#"+f)[0]||i("body").append(n),t.elem.next().hasClass(c)||(e.elemFile.wrap(a),t.elem.next("."+c).append(function(){var e=[];return layui.each(t.data,function(i,t){t="function"==typeof t?t():t,e.push('')}),e.join("")}()))},p.prototype.msg=function(e){return t.msg(e,{icon:2,shift:6})},p.prototype.isFile=function(){var e=this.config.elem[0];if(e)return"input"===e.tagName.toLocaleLowerCase()&&"file"===e.type},p.prototype.preview=function(e){var i=this;window.FileReader&&layui.each(i.chooseFiles,function(i,t){var n=new FileReader;n.readAsDataURL(t),n.onload=function(){e&&e(i,t,this.result)}})},p.prototype.upload=function(e,t){var n,o=this,l=o.config,r=o.elemFile[0],u=function(){var t=0,n=0,a=e||o.files||o.chooseFiles||r.files,u=function(){l.multiple&&t+n===o.fileLength&&"function"==typeof l.allDone&&l.allDone({total:o.fileLength,successful:t,aborted:n})};layui.each(a,function(e,a){var r=new FormData;r.append(l.field,a),layui.each(l.data,function(e,i){i="function"==typeof i?i():i,r.append(e,i)}),i.ajax({url:l.url,type:"post",data:r,contentType:!1,processData:!1,dataType:"json",headers:l.headers||{},success:function(i){t++,d(e,i),u()},error:function(){n++,o.msg("请求上传接口出现异常"),m(e),u()}})})},c=function(){var e=i("#"+f);o.elemFile.parent().submit(),clearInterval(p.timer),p.timer=setInterval(function(){var i,t=e.contents().find("body");try{i=t.text()}catch(n){o.msg("获取上传后的响应信息出现异常"),clearInterval(p.timer),m()}i&&(clearInterval(p.timer),t.html(""),d(0,i))},30)},d=function(e,i){if(o.elemFile.next("."+s).remove(),r.value="","object"!=typeof i)try{i=JSON.parse(i)}catch(t){return i={},o.msg("请对上传接口返回有效JSON")}"function"==typeof l.done&&l.done(i,e||0,function(e){o.upload(e)})},m=function(e){l.auto&&(r.value=""),"function"==typeof l.error&&l.error(e||0,function(e){o.upload(e)})},h=l.exts,v=function(){var i=[];return layui.each(e||o.chooseFiles,function(e,t){i.push(t.name)}),i}(),g={preview:function(e){o.preview(e)},upload:function(e,i){var t={};t[e]=i,o.upload(t)},pushFile:function(){return o.files=o.files||{},layui.each(o.chooseFiles,function(e,i){o.files[e]=i}),o.files},resetFile:function(e,i,t){var n=new File([i],t);o.files=o.files||{},o.files[e]=n}},y=function(){if("choose"!==t&&!l.auto||(l.choose&&l.choose(g),"choose"!==t))return l.before&&l.before(g),a.ie?a.ie>9?u():c():void u()};if(v=0===v.length?r.value.match(/[^\/\\]+\..+/g)||[]||"":v,0!==v.length){switch(l.accept){case"file":if(h&&!RegExp("\\w\\.("+h+")$","i").test(escape(v)))return o.msg("选择的文件中包含不支持的格式"),r.value="";break;case"video":if(!RegExp("\\w\\.("+(h||"avi|mp4|wma|rmvb|rm|flash|3gp|flv")+")$","i").test(escape(v)))return o.msg("选择的视频中包含不支持的格式"),r.value="";break;case"audio":if(!RegExp("\\w\\.("+(h||"mp3|wav|mid")+")$","i").test(escape(v)))return o.msg("选择的音频中包含不支持的格式"),r.value="";break;default:if(layui.each(v,function(e,i){RegExp("\\w\\.("+(h||"jpg|png|gif|bmp|jpeg$")+")","i").test(escape(i))||(n=!0)}),n)return o.msg("选择的图片中包含不支持的格式"),r.value=""}if(o.fileLength=function(){var i=0,t=e||o.files||o.chooseFiles||r.files;return layui.each(t,function(){i++}),i}(),l.number&&o.fileLength>l.number)return o.msg("同时最多只能上传的数量为:"+l.number);if(l.size>0&&!(a.ie&&a.ie<10)){var F;if(layui.each(o.chooseFiles,function(e,i){if(i.size>1024*l.size){var t=l.size/1024;t=t>=1?t.toFixed(2)+"MB":l.size+"KB",r.value="",F=t}}),F)return o.msg("文件不能超过"+F)}y()}},p.prototype.events=function(){var e=this,t=e.config,o=function(i){e.chooseFiles={},layui.each(i,function(i,t){var n=(new Date).getTime();e.chooseFiles[n+"-"+i]=t})},l=function(i,n){var a=e.elemFile,o=i.length>1?i.length+"个文件":(i[0]||{}).name||a[0].value.match(/[^\/\\]+\..+/g)||[]||"";a.next().hasClass(s)&&a.next().remove(),e.upload(null,"choose"),e.isFile()||t.choose||a.after(''+o+"")};t.elem.off("upload.start").on("upload.start",function(){var a=i(this),o=a.attr("lay-data");if(o)try{o=new Function("return "+o)(),e.config=i.extend({},t,o)}catch(l){n.error("Upload element property lay-data configuration item has a syntax error: "+o)}e.config.item=a,e.elemFile[0].click()}),a.ie&&a.ie<10||t.elem.off("upload.over").on("upload.over",function(){var e=i(this);e.attr("lay-over","")}).off("upload.leave").on("upload.leave",function(){var e=i(this);e.removeAttr("lay-over")}).off("upload.drop").on("upload.drop",function(n,a){var r=i(this),u=a.originalEvent.dataTransfer.files||[];r.removeAttr("lay-over"),o(u),t.auto?e.upload(u):l(u)}),e.elemFile.off("upload.change").on("upload.change",function(){var i=this.files||[];o(i),t.auto?e.upload():l(i)}),t.bindAction.off("upload.action").on("upload.action",function(){e.upload()}),t.elem.data("haveEvents")||(e.elemFile.on("change",function(){i(this).trigger("upload.change")}),t.elem.on("click",function(){e.isFile()||i(this).trigger("upload.start")}),t.drag&&t.elem.on("dragover",function(e){e.preventDefault(),i(this).trigger("upload.over")}).on("dragleave",function(e){i(this).trigger("upload.leave")}).on("drop",function(e){e.preventDefault(),i(this).trigger("upload.drop",e)}),t.bindAction.on("click",function(){i(this).trigger("upload.action")}),t.elem.data("haveEvents",!0))},o.render=function(e){var i=new p(e);return l.call(i)},e(r,o)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/lay/modules/util.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;layui.define("jquery",function(t){"use strict";var e=layui.$,i={fixbar:function(t){var i,a,n="layui-fixbar",r="layui-fixbar-top",o=e(document),l=e("body");t=e.extend({showHeight:200},t),t.bar1=t.bar1===!0?"":t.bar1,t.bar2=t.bar2===!0?"":t.bar2,t.bgcolor=t.bgcolor?"background-color:"+t.bgcolor:"";var c=[t.bar1,t.bar2,""],g=e(['',t.bar1?'- '+c[0]+"
":"",t.bar2?'- '+c[1]+"
":"",'- '+c[2]+"
","
"].join("")),s=g.find("."+r),u=function(){var e=o.scrollTop();e>=t.showHeight?i||(s.show(),i=1):i&&(s.hide(),i=0)};e("."+n)[0]||("object"==typeof t.css&&g.css(t.css),l.append(g),u(),g.find("li").on("click",function(){var i=e(this),a=i.attr("lay-type");"top"===a&&e("html,body").animate({scrollTop:0},200),t.click&&t.click.call(this,a)}),o.on("scroll",function(){clearTimeout(a),a=setTimeout(function(){u()},100)}))},countdown:function(t,e,i){var a=this,n="function"==typeof e,r=new Date(t).getTime(),o=new Date(!e||n?(new Date).getTime():e).getTime(),l=r-o,c=[Math.floor(l/864e5),Math.floor(l/36e5)%24,Math.floor(l/6e4)%60,Math.floor(l/1e3)%60];n&&(i=e);var g=setTimeout(function(){a.countdown(t,o+1e3,i)},1e3);return i&&i(l>0?c:[0,0,0,0],e,g),l<=0&&clearTimeout(g),g},timeAgo:function(t,e){var i=this,a=[[],[]],n=(new Date).getTime()-new Date(t).getTime();return n>6912e5?(n=new Date(t),a[0][0]=i.digit(n.getFullYear(),4),a[0][1]=i.digit(n.getMonth()+1),a[0][2]=i.digit(n.getDate()),e||(a[1][0]=i.digit(n.getHours()),a[1][1]=i.digit(n.getMinutes()),a[1][2]=i.digit(n.getSeconds())),a[0].join("-")+" "+a[1].join(":")):n>=864e5?(n/1e3/60/60/24|0)+"天前":n>=36e5?(n/1e3/60/60|0)+"小时前":n>=12e4?(n/1e3/60|0)+"分钟前":n<0?"未来":"刚刚"},digit:function(t,e){var i="";t=String(t),e=e||2;for(var a=t.length;a/g,">").replace(/'/g,"'").replace(/"/g,""")}};!function(t,e,i){"$:nomunge";function a(){n=e[l](function(){r.each(function(){var e=t(this),i=e.width(),a=e.height(),n=t.data(this,g);(i!==n.w||a!==n.h)&&e.trigger(c,[n.w=i,n.h=a])}),a()},o[s])}var n,r=t([]),o=t.resize=t.extend(t.resize,{}),l="setTimeout",c="resize",g=c+"-special-event",s="delay",u="throttleWindow";o[s]=250,o[u]=!0,t.event.special[c]={setup:function(){if(!o[u]&&this[l])return!1;var e=t(this);r=r.add(e),t.data(this,g,{w:e.width(),h:e.height()}),1===r.length&&a()},teardown:function(){if(!o[u]&&this[l])return!1;var e=t(this);r=r.not(e),e.removeData(g),r.length||clearTimeout(n)},add:function(e){function a(e,a,r){var o=t(this),l=t.data(this,g)||{};l.w=a!==i?a:o.width(),l.h=r!==i?r:o.height(),n.apply(this,arguments)}if(!o[u]&&this[l])return!1;var n;return t.isFunction(e)?(n=e,a):(n=e.handler,void(e.handler=a))}}}(e,window),t("util",i)});
--------------------------------------------------------------------------------
/src/main/resources/static/layui/layui.js:
--------------------------------------------------------------------------------
1 | /** layui-v2.4.5 MIT License By https://www.layui.com */
2 | ;!function(e){"use strict";var t=document,o={modules:{},status:{},timeout:10,event:{}},n=function(){this.v="2.4.5"},r=function(){var e=t.currentScript?t.currentScript.src:function(){for(var e,o=t.scripts,n=o.length-1,r=n;r>0;r--)if("interactive"===o[r].readyState){e=o[r].src;break}return e||o[n].src}();return e.substring(0,e.lastIndexOf("/")+1)}(),i=function(t){e.console&&console.error&&console.error("Layui hint: "+t)},a="undefined"!=typeof opera&&"[object Opera]"===opera.toString(),u={layer:"modules/layer",laydate:"modules/laydate",laypage:"modules/laypage",laytpl:"modules/laytpl",layim:"modules/layim",layedit:"modules/layedit",form:"modules/form",upload:"modules/upload",tree:"modules/tree",table:"modules/table",element:"modules/element",rate:"modules/rate",colorpicker:"modules/colorpicker",slider:"modules/slider",carousel:"modules/carousel",flow:"modules/flow",util:"modules/util",code:"modules/code",jquery:"modules/jquery",mobile:"modules/mobile","layui.all":"../layui.all"};n.prototype.cache=o,n.prototype.define=function(e,t){var n=this,r="function"==typeof e,i=function(){var e=function(e,t){layui[e]=t,o.status[e]=!0};return"function"==typeof t&&t(function(n,r){e(n,r),o.callback[n]=function(){t(e)}}),this};return r&&(t=e,e=[]),layui["layui.all"]||!layui["layui.all"]&&layui["layui.mobile"]?i.call(n):(n.use(e,i),n)},n.prototype.use=function(e,n,l){function s(e,t){var n="PLaySTATION 3"===navigator.platform?/^complete$/:/^(complete|loaded)$/;("load"===e.type||n.test((e.currentTarget||e.srcElement).readyState))&&(o.modules[f]=t,d.removeChild(v),function r(){return++m>1e3*o.timeout/4?i(f+" is not a valid module"):void(o.status[f]?c():setTimeout(r,4))}())}function c(){l.push(layui[f]),e.length>1?y.use(e.slice(1),n,l):"function"==typeof n&&n.apply(layui,l)}var y=this,p=o.dir=o.dir?o.dir:r,d=t.getElementsByTagName("head")[0];e="string"==typeof e?[e]:e,window.jQuery&&jQuery.fn.on&&(y.each(e,function(t,o){"jquery"===o&&e.splice(t,1)}),layui.jquery=layui.$=jQuery);var f=e[0],m=0;if(l=l||[],o.host=o.host||(p.match(/\/\/([\s\S]+?)\//)||["//"+location.host+"/"])[0],0===e.length||layui["layui.all"]&&u[f]||!layui["layui.all"]&&layui["layui.mobile"]&&u[f])return c(),y;if(o.modules[f])!function g(){return++m>1e3*o.timeout/4?i(f+" is not a valid module"):void("string"==typeof o.modules[f]&&o.status[f]?c():setTimeout(g,4))}();else{var v=t.createElement("script"),h=(u[f]?p+"lay/":/^\{\/\}/.test(y.modules[f])?"":o.base||"")+(y.modules[f]||f)+".js";h=h.replace(/^\{\/\}/,""),v.async=!0,v.charset="utf-8",v.src=h+function(){var e=o.version===!0?o.v||(new Date).getTime():o.version||"";return e?"?v="+e:""}(),d.appendChild(v),!v.attachEvent||v.attachEvent.toString&&v.attachEvent.toString().indexOf("[native code")<0||a?v.addEventListener("load",function(e){s(e,h)},!1):v.attachEvent("onreadystatechange",function(e){s(e,h)}),o.modules[f]=h}return y},n.prototype.getStyle=function(t,o){var n=t.currentStyle?t.currentStyle:e.getComputedStyle(t,null);return n[n.getPropertyValue?"getPropertyValue":"getAttribute"](o)},n.prototype.link=function(e,n,r){var a=this,u=t.createElement("link"),l=t.getElementsByTagName("head")[0];"string"==typeof n&&(r=n);var s=(r||e).replace(/\.|\//g,""),c=u.id="layuicss-"+s,y=0;return u.rel="stylesheet",u.href=e+(o.debug?"?v="+(new Date).getTime():""),u.media="all",t.getElementById(c)||l.appendChild(u),"function"!=typeof n?a:(function p(){return++y>1e3*o.timeout/100?i(e+" timeout"):void(1989===parseInt(a.getStyle(t.getElementById(c),"width"))?function(){n()}():setTimeout(p,100))}(),a)},o.callback={},n.prototype.factory=function(e){if(layui[e])return"function"==typeof o.callback[e]?o.callback[e]:null},n.prototype.addcss=function(e,t,n){return layui.link(o.dir+"css/"+e,t,n)},n.prototype.img=function(e,t,o){var n=new Image;return n.src=e,n.complete?t(n):(n.onload=function(){n.onload=null,"function"==typeof t&&t(n)},void(n.onerror=function(e){n.onerror=null,"function"==typeof o&&o(e)}))},n.prototype.config=function(e){e=e||{};for(var t in e)o[t]=e[t];return this},n.prototype.modules=function(){var e={};for(var t in u)e[t]=u[t];return e}(),n.prototype.extend=function(e){var t=this;e=e||{};for(var o in e)t[o]||t.modules[o]?i("模块名 "+o+" 已被占用"):t.modules[o]=e[o];return t},n.prototype.router=function(e){var t=this,e=e||location.hash,o={path:[],search:{},hash:(e.match(/[^#](#.*$)/)||[])[1]||""};return/^#\//.test(e)?(e=e.replace(/^#\//,""),o.href="/"+e,e=e.replace(/([^#])(#.*$)/,"$1").split("/")||[],t.each(e,function(e,t){/^\w+=/.test(t)?function(){t=t.split("="),o.search[t[0]]=t[1]}():o.path.push(t)}),o):o},n.prototype.data=function(t,o,n){if(t=t||"layui",n=n||localStorage,e.JSON&&e.JSON.parse){if(null===o)return delete n[t];o="object"==typeof o?o:{key:o};try{var r=JSON.parse(n[t])}catch(i){var r={}}return"value"in o&&(r[o.key]=o.value),o.remove&&delete r[o.key],n[t]=JSON.stringify(r),o.key?r[o.key]:r}},n.prototype.sessionData=function(e,t){return this.data(e,t,sessionStorage)},n.prototype.device=function(t){var o=navigator.userAgent.toLowerCase(),n=function(e){var t=new RegExp(e+"/([^\\s\\_\\-]+)");return e=(o.match(t)||[])[1],e||!1},r={os:function(){return/windows/.test(o)?"windows":/linux/.test(o)?"linux":/iphone|ipod|ipad|ios/.test(o)?"ios":/mac/.test(o)?"mac":void 0}(),ie:function(){return!!(e.ActiveXObject||"ActiveXObject"in e)&&((o.match(/msie\s(\d+)/)||[])[1]||"11")}(),weixin:n("micromessenger")};return t&&!r[t]&&(r[t]=n(t)),r.android=/android/.test(o),r.ios="ios"===r.os,r},n.prototype.hint=function(){return{error:i}},n.prototype.each=function(e,t){var o,n=this;if("function"!=typeof t)return n;if(e=e||[],e.constructor===Object){for(o in e)if(t.call(e[o],o,e[o]))break}else for(o=0;oi?1:r
2 |
3 |
4 |
5 | Title
6 |
7 |
8 |
9 |
10 |
41 |
42 |
--------------------------------------------------------------------------------
/src/test/java/com/bektz/dataplatformsoar/DataPlatformSoarApplicationTests.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar;
2 |
3 | import org.junit.Test;
4 | import org.junit.runner.RunWith;
5 | import org.springframework.boot.test.context.SpringBootTest;
6 | import org.springframework.test.context.junit4.SpringRunner;
7 |
8 | @RunWith(SpringRunner.class)
9 | @SpringBootTest
10 | public class DataPlatformSoarApplicationTests {
11 |
12 | @Test
13 | public void contextLoads() {
14 | }
15 |
16 | }
17 |
--------------------------------------------------------------------------------
/src/test/java/com/bektz/dataplatformsoar/MysqlTest.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar;
2 |
3 | import com.alibaba.druid.pool.DruidDataSource;
4 | import com.alibaba.druid.pool.DruidPooledConnection;
5 | import com.alibaba.druid.util.JdbcUtils;
6 | import org.junit.Assert;
7 | import org.junit.Before;
8 | import org.junit.Test;
9 |
10 | import java.sql.SQLException;
11 | import java.util.List;
12 | import java.util.Map;
13 |
14 | public class MysqlTest {
15 |
16 |
17 | DruidDataSource dataSource = null;
18 |
19 | @Before
20 | public void setup() throws Exception {
21 | dataSource = new DruidDataSource();
22 | dataSource.setUsername("root");
23 | dataSource.setUrl("jdbc:mysql://127.0.0.1:3306/local");
24 | dataSource.setPassword("root");
25 | dataSource.setMaxActive(2);
26 | dataSource.setDriverClassName("com.mysql.jdbc.Driver");
27 | DruidPooledConnection connection = dataSource.getConnection(1000);
28 | }
29 |
30 |
31 | @Test
32 | public void mysqlTest() throws SQLException {
33 | boolean asyncInit = dataSource.isAsyncInit();
34 | System.out.println(asyncInit);
35 | // DruidPooledConnection connection = dataSource.getConnection();
36 | // Statement statement = connection.createStatement();
37 | // boolean execute = statement.execute("select 1");
38 | // System.out.println(execute);
39 | }
40 |
41 |
42 | @Test
43 | public void jdbcUtilsTest() throws Exception {
44 | List> show_tables = JdbcUtils.executeQuery(dataSource, "select * from test");
45 | Assert.assertTrue(show_tables.size() > 0);
46 | }
47 |
48 | }
49 |
--------------------------------------------------------------------------------
/src/test/java/com/bektz/dataplatformsoar/service/SoarServiceTest.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.service;
2 |
3 | import com.bektz.dataplatformsoar.req.SqlVerifyReq;
4 | import com.bektz.dataplatformsoar.resp.SqlVerifyResp;
5 | import org.junit.Assert;
6 | import org.junit.jupiter.api.Test;
7 | import org.junit.runner.RunWith;
8 | import org.springframework.beans.factory.annotation.Autowired;
9 | import org.springframework.boot.test.context.SpringBootTest;
10 | import org.springframework.test.context.junit4.SpringRunner;
11 |
12 | import static org.junit.jupiter.api.Assertions.*;
13 |
14 | @RunWith(SpringRunner.class)
15 | @SpringBootTest
16 | class SoarServiceTest {
17 | @Autowired
18 | private SoarService soarService;
19 |
20 | @Test
21 | void verify() {
22 | SqlVerifyResp verify = soarService.verify(SqlVerifyReq.builder().sql("CREATE TABLE `testfyl` (`id` int(11) unsigned NOT NULL AUTO_INCREMENT,`test` varchar(20) DEFAULT NULL,`dfa` varchar(20) DEFAULT NULL,`hhh` varchar(20) DEFAULT NULL,PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULTCHARSET=utf8mb4;").build());
23 | Assert.assertNotNull(verify);
24 | }
25 |
26 | }
--------------------------------------------------------------------------------
/src/test/java/com/bektz/dataplatformsoar/sqlparser/DruidSqlParserTest.java:
--------------------------------------------------------------------------------
1 | package com.bektz.dataplatformsoar.sqlparser;
2 |
3 | import com.bektz.dataplatformsoar.sqlparser.druid.DruidSqlParser;
4 | import org.junit.jupiter.api.Test;
5 |
6 | import java.util.Map;
7 |
8 | class DruidSqlParserTest {
9 |
10 |
11 | String[] sqls = new String[]{
12 | // "SELECT * ,username FROM MY_TABLE1, MY_TABLE2, (SELECT username FROM MY_TABLE3) LEFT OUTER JOIN MY_TABLE4 \" +\n" +
13 | // " \" WHERE ID = (SELECT MAX(ID) FROM MY_TABLE5) AND ID2 IN (SELECT * FROM MY_TABLE6)",
14 | "select a.name c from (select alert.name from alert inner join user on alert.id = user.id) a",
15 | "select a.m cc , a.nnn from (select c.mobile as m ,c.nnn nnn from card c where c.id = 1) a",
16 | "select u.name,u.no from (select u.name name ,c.card_no no from user u inner join card c on u.id = c.id) u",
17 | "select * from (select a,avg(*) b from alert) c",
18 | "select a.na a from (select u.name as na , u.age from user u inner join alert al on alert.id = user.id) a",
19 | "select a.m as cc from (select c.mobile as m from (select mobile from card) c) a",
20 | "select * from firekylin.alert",
21 | "select st.sname,st.sno from student st join (select sc.sno sn,count(sc.cno) cou from sc group by sc.sno) scs on st.sno=scs.sn where scs.cou <(select count(cno) from course);",
22 | "select distinct st.sno, sname\n" +
23 | " from student st\n" +
24 | " join sc sc on (st.sno = sc.sno)\n" +
25 | " where sc.cno in (select cno from sc where sno = 's001') and sc.sno<>'s001';"
26 | };
27 |
28 | @Test
29 | void parserRealMetaData() {
30 | DruidSqlParser druidSqlParser = new DruidSqlParser();
31 | for (String sql : sqls) {
32 | Map columnItems = druidSqlParser.parserSql(sql);
33 | System.out.println(columnItems);
34 | }
35 | }
36 |
37 |
38 | @Test
39 | void parserTest() {
40 | String sql = "select distinct st.sno, sname\n" +
41 | " from student st\n" +
42 | " join sc sc on (st.sno = sc.sno)\n" +
43 | " where sc.cno in (select cno from sc where sno = 's001') and sc.sno<>'s001';";
44 | DruidSqlParser druidSqlParser = new DruidSqlParser();
45 | Map columnItems = druidSqlParser.parserSql(sql);
46 | System.out.println(columnItems);
47 |
48 | }
49 | }
--------------------------------------------------------------------------------