切换到jackson #10
26
.github/workflows/build.yml
vendored
Normal file
26
.github/workflows/build.yml
vendored
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# This workflow will build a package using Maven and then publish it to GitHub packages when a release is created
|
||||||
|
# For more information see: https://github.com/actions/setup-java#apache-maven-with-a-settings-path
|
||||||
|
|
||||||
|
name: Build
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
KEY: ${{ secrets.WEB_HOOK_ACCESS_KEY }}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Set up JDK 1.8
|
||||||
|
uses: actions/setup-java@v1
|
||||||
|
with:
|
||||||
|
java-version: 1.8
|
||||||
|
|
||||||
|
- name: Deploy
|
||||||
|
run: mvn -B test --file pom.xml && curl http://106.15.205.190:2271/hook?access_key=$KEY
|
||||||
47
.github/workflows/deplay.yml
vendored
47
.github/workflows/deplay.yml
vendored
@@ -1,47 +0,0 @@
|
|||||||
# This workflow will build a package using Maven and then publish it to GitHub packages when a release is created
|
|
||||||
# For more information see: https://github.com/actions/setup-java#apache-maven-with-a-settings-path
|
|
||||||
|
|
||||||
name: Deplay
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
env:
|
|
||||||
APPLICATION_PROPERTIES_TEST: ${{ secrets.APPLICATION_PROPERTIES_TEST }}
|
|
||||||
APPLICATION_PROPERTIES_PROD: ${{ secrets.APPLICATION_PROPERTIES_PROD }}
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v2
|
|
||||||
- name: Set up JDK 1.8
|
|
||||||
uses: actions/setup-java@v1
|
|
||||||
with:
|
|
||||||
java-version: 1.8
|
|
||||||
|
|
||||||
- name: Build jar file
|
|
||||||
run: echo $APPLICATION_PROPERTIES_PROD|base64 -d> src/main/resources/application-prod.properties && mvn -B package --file pom.xml
|
|
||||||
|
|
||||||
- name: SCP
|
|
||||||
uses: appleboy/scp-action@master
|
|
||||||
with:
|
|
||||||
host: ${{ secrets.SSH_HOST }}
|
|
||||||
username: ${{ secrets.SSH_USERNAME }}
|
|
||||||
password: ${{ secrets.SSH_PASSWORD }}
|
|
||||||
port: ${{ secrets.SSH_PORT }}
|
|
||||||
source: "target/blog-0.0.1-SNAPSHOT.jar"
|
|
||||||
target: "/www/wwwroot/api.celess.cn"
|
|
||||||
|
|
||||||
- name: Run SSH command
|
|
||||||
uses: garygrossgarten/github-action-ssh@v0.5.0
|
|
||||||
with:
|
|
||||||
command: cd /www/wwwroot/api.celess.cn && bash build.sh
|
|
||||||
host: ${{ secrets.SSH_HOST }}
|
|
||||||
username: ${{ secrets.SSH_USERNAME }}
|
|
||||||
password: ${{ secrets.SSH_PASSWORD }}
|
|
||||||
port: ${{ secrets.SSH_PORT }}
|
|
||||||
|
|
||||||
10
.github/workflows/test.yml
vendored
10
.github/workflows/test.yml
vendored
@@ -1,7 +1,7 @@
|
|||||||
# This workflow will build a package using Maven and then publish it to GitHub packages when a release is created
|
# This workflow will build a package using Maven and then publish it to GitHub packages when a release is created
|
||||||
# For more information see: https://github.com/actions/setup-java#apache-maven-with-a-settings-path
|
# For more information see: https://github.com/actions/setup-java#apache-maven-with-a-settings-path
|
||||||
|
|
||||||
name: pr Test
|
name: Test
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
@@ -12,9 +12,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
env:
|
# env:
|
||||||
APPLICATION_PROPERTIES_TEST: ${{ secrets.APPLICATION_PROPERTIES_TEST }}
|
# APPLICATION_PROPERTIES_TEST: ${{ secrets.APPLICATION_PROPERTIES_TEST }}
|
||||||
APPLICATION_PROPERTIES_PROD: ${{ secrets.APPLICATION_PROPERTIES_PROD }}
|
# APPLICATION_PROPERTIES_PROD: ${{ secrets.APPLICATION_PROPERTIES_PROD }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
@@ -24,6 +24,6 @@ jobs:
|
|||||||
java-version: 1.8
|
java-version: 1.8
|
||||||
|
|
||||||
- name: Test
|
- name: Test
|
||||||
run: echo $APPLICATION_PROPERTIES_TEST|base64 -d > src/main/resources/application-test.properties && mvn -B test --file pom.xml
|
run: mvn -B test --file pom.xml
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -4,7 +4,5 @@
|
|||||||
target/
|
target/
|
||||||
|
|
||||||
# 本地项目的私有文件
|
# 本地项目的私有文件
|
||||||
back-end/blog-dev.sql
|
|
||||||
src/main/resources/application-dev.properties
|
src/main/resources/application-dev.properties
|
||||||
src/main/resources/application-prod.properties
|
src/main/resources/application-prod.properties
|
||||||
src/main/resources/application-test.properties
|
|
||||||
|
|||||||
7
.mvn/wrapper/MavenWrapperDownloader.java
vendored
7
.mvn/wrapper/MavenWrapperDownloader.java
vendored
@@ -1,11 +1,11 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright 2012-2019 the original author or authors.
|
* Copyright 2007-present the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
* You may obtain a copy of the License at
|
* You may obtain a copy of the License at
|
||||||
*
|
*
|
||||||
* https://www.apache.org/licenses/LICENSE-2.0
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
*
|
*
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
@@ -13,7 +13,6 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import java.net.*;
|
import java.net.*;
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
import java.nio.channels.*;
|
import java.nio.channels.*;
|
||||||
@@ -21,7 +20,7 @@ import java.util.Properties;
|
|||||||
|
|
||||||
public class MavenWrapperDownloader {
|
public class MavenWrapperDownloader {
|
||||||
|
|
||||||
private static final String WRAPPER_VERSION = "0.5.5";
|
private static final String WRAPPER_VERSION = "0.5.6";
|
||||||
/**
|
/**
|
||||||
* Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided.
|
* Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided.
|
||||||
*/
|
*/
|
||||||
|
|||||||
BIN
.mvn/wrapper/maven-wrapper.jar
vendored
BIN
.mvn/wrapper/maven-wrapper.jar
vendored
Binary file not shown.
4
.mvn/wrapper/maven-wrapper.properties
vendored
4
.mvn/wrapper/maven-wrapper.properties
vendored
@@ -1,2 +1,2 @@
|
|||||||
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.2/apache-maven-3.6.2-bin.zip
|
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip
|
||||||
wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.5/maven-wrapper-0.5.5.jar
|
wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar
|
||||||
|
|||||||
2
LICENSE
2
LICENSE
@@ -1,6 +1,6 @@
|
|||||||
MIT License
|
MIT License
|
||||||
|
|
||||||
Copyright (c) 2019 小海
|
Copyright (c) 2020 禾几海
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
|||||||
32
README.md
32
README.md
@@ -1,11 +1,19 @@
|
|||||||
# 小海博客后端管理系统
|
<h1 align="center">
|
||||||
   [](https://www.celess.cn)
|
小海博客后端管理系统
|
||||||
## 基于Springboot的后端博客管理系统
|
</h1>
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
|
||||||
|
基于Springboot的后端博客管理系统
|
||||||
|
|
||||||
|
[](https://github.com/xiaohai2271/blog-backEnd)
|
||||||
|
[](https://github.com/xiaohai2271/blog-backEnd)
|
||||||
|
[](https://github.com/xiaohai2271/blog-backEnd)
|
||||||
|
[](https://www.celess.cn)
|
||||||
|
|
||||||
### 主要使用的技术
|
</div>
|
||||||
|
|
||||||
|
## 🎈主要使用的技术
|
||||||
|
|
||||||
| 使用的技术 | 名称 | 版本 |
|
| 使用的技术 | 名称 | 版本 |
|
||||||
| :--------------: | :--------: | :-----------: |
|
| :--------------: | :--------: | :-----------: |
|
||||||
@@ -20,16 +28,14 @@
|
|||||||
| 缓存(线上环境) | redis | 3.0.6 |
|
| 缓存(线上环境) | redis | 3.0.6 |
|
||||||
| 数据库 |mysql |5.7 |
|
| 数据库 |mysql |5.7 |
|
||||||
|
|
||||||
### 接口文档
|
## 🔨如何构建
|
||||||
|
详情参照[Build](./doc/Build.md)文档
|
||||||
|
|
||||||
项目采用swagger2,接口文档自动生成,具体为 http://ip:端口/swagger-ui.html
|
|
||||||
|
|
||||||
### 📝TODO
|
## 📒接口文档
|
||||||
|
项目采用swagger2,接口文档自动生成,具体为 http://ip:port/doc.html
|
||||||
|
|
||||||
- [x] 密码重置
|
或者参照[离线API文档](./doc/API.md)
|
||||||
- [x] 信息修改
|
|
||||||
- [ ] 接入qq登录
|
|
||||||
|
|
||||||
### 📌FIXME
|
## ☀授权协议
|
||||||
|
[MIT](./LICENSE)
|
||||||
- [ ] `/write` 图片上传的跨域问题
|
|
||||||
4077
doc/API.md
Normal file
4077
doc/API.md
Normal file
File diff suppressed because it is too large
Load Diff
32
doc/Build.md
Normal file
32
doc/Build.md
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
# 如何构建
|
||||||
|
### 1. 构建前准备
|
||||||
|
- 安装jdk
|
||||||
|
- 安装maven(也可使使用项目中的maven wrapper)
|
||||||
|
- 安装mysql
|
||||||
|
- 安装redis
|
||||||
|
- 获取七牛云的AccessKey/SecretKey (个人中心-密钥管理) [七牛云官网](https://www.qiniu.com/)
|
||||||
|
|
||||||
|
### 2. 拉取项目到本地
|
||||||
|
``` shell script
|
||||||
|
git clone https://github.com/xiaohai2271/blog-backEnd.git
|
||||||
|
或 git clone git@github.com:xiaohai2271/blog-backEnd.git
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. maven构建
|
||||||
|
```shell script
|
||||||
|
mvn package
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 运行
|
||||||
|
```shell script
|
||||||
|
java -jar target/blog-0.0.1-SNAPSHOT.jar
|
||||||
|
```
|
||||||
|
|
||||||
|
### 5. 其他命令
|
||||||
|
```shell script
|
||||||
|
mvn clean # 清理项目资源
|
||||||
|
mvn clean package # 清理项目资源并重新打包
|
||||||
|
mvn clean package -DskipTest # 清理项目资源,重新打包并跳过测试
|
||||||
|
mvn test # 运行测试
|
||||||
|
..... #待添加
|
||||||
|
```
|
||||||
98
mvnw
vendored
98
mvnw
vendored
@@ -8,7 +8,7 @@
|
|||||||
# "License"); you may not use this file except in compliance
|
# "License"); you may not use this file except in compliance
|
||||||
# with the License. You may obtain a copy of the License at
|
# with the License. You may obtain a copy of the License at
|
||||||
#
|
#
|
||||||
# https://www.apache.org/licenses/LICENSE-2.0
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
#
|
#
|
||||||
# Unless required by applicable law or agreed to in writing,
|
# Unless required by applicable law or agreed to in writing,
|
||||||
# software distributed under the License is distributed on an
|
# software distributed under the License is distributed on an
|
||||||
@@ -19,7 +19,7 @@
|
|||||||
# ----------------------------------------------------------------------------
|
# ----------------------------------------------------------------------------
|
||||||
|
|
||||||
# ----------------------------------------------------------------------------
|
# ----------------------------------------------------------------------------
|
||||||
# Maven2 Start Up Batch script
|
# Maven Start Up Batch script
|
||||||
#
|
#
|
||||||
# Required ENV vars:
|
# Required ENV vars:
|
||||||
# ------------------
|
# ------------------
|
||||||
@@ -47,19 +47,18 @@ if [ -z "$MAVEN_SKIP_RC" ]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
# OS specific support. $var _must_ be set to either true or false.
|
# OS specific support. $var _must_ be set to either true or false.
|
||||||
cygwin=false
|
cygwin=false;
|
||||||
darwin=false
|
darwin=false;
|
||||||
mingw=false
|
mingw=false
|
||||||
case "$(uname)" in
|
case "`uname`" in
|
||||||
CYGWIN*) cygwin=true ;;
|
CYGWIN*) cygwin=true ;;
|
||||||
MINGW*) mingw=true;;
|
MINGW*) mingw=true;;
|
||||||
Darwin*)
|
Darwin*) darwin=true
|
||||||
darwin=true
|
|
||||||
# Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home
|
# Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home
|
||||||
# See https://developer.apple.com/library/mac/qa/qa1170/_index.html
|
# See https://developer.apple.com/library/mac/qa/qa1170/_index.html
|
||||||
if [ -z "$JAVA_HOME" ]; then
|
if [ -z "$JAVA_HOME" ]; then
|
||||||
if [ -x "/usr/libexec/java_home" ]; then
|
if [ -x "/usr/libexec/java_home" ]; then
|
||||||
export JAVA_HOME="$(/usr/libexec/java_home)"
|
export JAVA_HOME="`/usr/libexec/java_home`"
|
||||||
else
|
else
|
||||||
export JAVA_HOME="/Library/Java/Home"
|
export JAVA_HOME="/Library/Java/Home"
|
||||||
fi
|
fi
|
||||||
@@ -69,7 +68,7 @@ esac
|
|||||||
|
|
||||||
if [ -z "$JAVA_HOME" ] ; then
|
if [ -z "$JAVA_HOME" ] ; then
|
||||||
if [ -r /etc/gentoo-release ] ; then
|
if [ -r /etc/gentoo-release ] ; then
|
||||||
JAVA_HOME=$(java-config --jre-home)
|
JAVA_HOME=`java-config --jre-home`
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -79,21 +78,21 @@ if [ -z "$M2_HOME" ]; then
|
|||||||
|
|
||||||
# need this for relative symlinks
|
# need this for relative symlinks
|
||||||
while [ -h "$PRG" ] ; do
|
while [ -h "$PRG" ] ; do
|
||||||
ls=$(ls -ld "$PRG")
|
ls=`ls -ld "$PRG"`
|
||||||
link=$(expr "$ls" : '.*-> \(.*\)$')
|
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||||
if expr "$link" : '/.*' > /dev/null; then
|
if expr "$link" : '/.*' > /dev/null; then
|
||||||
PRG="$link"
|
PRG="$link"
|
||||||
else
|
else
|
||||||
PRG="$(dirname "$PRG")/$link"
|
PRG="`dirname "$PRG"`/$link"
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
saveddir=$(pwd)
|
saveddir=`pwd`
|
||||||
|
|
||||||
M2_HOME=$(dirname "$PRG")/..
|
M2_HOME=`dirname "$PRG"`/..
|
||||||
|
|
||||||
# make it fully qualified
|
# make it fully qualified
|
||||||
M2_HOME=$(cd "$M2_HOME" && pwd)
|
M2_HOME=`cd "$M2_HOME" && pwd`
|
||||||
|
|
||||||
cd "$saveddir"
|
cd "$saveddir"
|
||||||
# echo Using m2 at $M2_HOME
|
# echo Using m2 at $M2_HOME
|
||||||
@@ -102,41 +101,35 @@ fi
|
|||||||
# For Cygwin, ensure paths are in UNIX format before anything is touched
|
# For Cygwin, ensure paths are in UNIX format before anything is touched
|
||||||
if $cygwin ; then
|
if $cygwin ; then
|
||||||
[ -n "$M2_HOME" ] &&
|
[ -n "$M2_HOME" ] &&
|
||||||
M2_HOME=$(cygpath --unix "$M2_HOME")
|
M2_HOME=`cygpath --unix "$M2_HOME"`
|
||||||
[ -n "$JAVA_HOME" ] &&
|
[ -n "$JAVA_HOME" ] &&
|
||||||
JAVA_HOME=$(cygpath --unix "$JAVA_HOME")
|
JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
|
||||||
[ -n "$CLASSPATH" ] &&
|
[ -n "$CLASSPATH" ] &&
|
||||||
CLASSPATH=$(cygpath --path --unix "$CLASSPATH")
|
CLASSPATH=`cygpath --path --unix "$CLASSPATH"`
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# For Mingw, ensure paths are in UNIX format before anything is touched
|
# For Mingw, ensure paths are in UNIX format before anything is touched
|
||||||
if $mingw ; then
|
if $mingw ; then
|
||||||
[ -n "$M2_HOME" ] &&
|
[ -n "$M2_HOME" ] &&
|
||||||
M2_HOME="$( (
|
M2_HOME="`(cd "$M2_HOME"; pwd)`"
|
||||||
cd "$M2_HOME"
|
|
||||||
pwd
|
|
||||||
))"
|
|
||||||
[ -n "$JAVA_HOME" ] &&
|
[ -n "$JAVA_HOME" ] &&
|
||||||
JAVA_HOME="$( (
|
JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`"
|
||||||
cd "$JAVA_HOME"
|
|
||||||
pwd
|
|
||||||
))"
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ -z "$JAVA_HOME" ]; then
|
if [ -z "$JAVA_HOME" ]; then
|
||||||
javaExecutable="$(which javac)"
|
javaExecutable="`which javac`"
|
||||||
if [ -n "$javaExecutable" ] && ! [ "$(expr \"$javaExecutable\" : '\([^ ]*\)')" = "no" ]; then
|
if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then
|
||||||
# readlink(1) is not available as standard on Solaris 10.
|
# readlink(1) is not available as standard on Solaris 10.
|
||||||
readLink=$(which readlink)
|
readLink=`which readlink`
|
||||||
if [ ! $(expr "$readLink" : '\([^ ]*\)') = "no" ]; then
|
if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then
|
||||||
if $darwin ; then
|
if $darwin ; then
|
||||||
javaHome="$(dirname \"$javaExecutable\")"
|
javaHome="`dirname \"$javaExecutable\"`"
|
||||||
javaExecutable="$(cd \"$javaHome\" && pwd -P)/javac"
|
javaExecutable="`cd \"$javaHome\" && pwd -P`/javac"
|
||||||
else
|
else
|
||||||
javaExecutable="$(readlink -f \"$javaExecutable\")"
|
javaExecutable="`readlink -f \"$javaExecutable\"`"
|
||||||
fi
|
fi
|
||||||
javaHome="$(dirname \"$javaExecutable\")"
|
javaHome="`dirname \"$javaExecutable\"`"
|
||||||
javaHome=$(expr "$javaHome" : '\(.*\)/bin')
|
javaHome=`expr "$javaHome" : '\(.*\)/bin'`
|
||||||
JAVA_HOME="$javaHome"
|
JAVA_HOME="$javaHome"
|
||||||
export JAVA_HOME
|
export JAVA_HOME
|
||||||
fi
|
fi
|
||||||
@@ -152,7 +145,7 @@ if [ -z "$JAVACMD" ]; then
|
|||||||
JAVACMD="$JAVA_HOME/bin/java"
|
JAVACMD="$JAVA_HOME/bin/java"
|
||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
JAVACMD="$(which java)"
|
JAVACMD="`which java`"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -172,7 +165,8 @@ CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher
|
|||||||
# first directory with .mvn subdirectory is considered project base directory
|
# first directory with .mvn subdirectory is considered project base directory
|
||||||
find_maven_basedir() {
|
find_maven_basedir() {
|
||||||
|
|
||||||
if [ -z "$1" ]; then
|
if [ -z "$1" ]
|
||||||
|
then
|
||||||
echo "Path not specified to find_maven_basedir"
|
echo "Path not specified to find_maven_basedir"
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
@@ -186,10 +180,7 @@ find_maven_basedir() {
|
|||||||
fi
|
fi
|
||||||
# workaround for JBEAP-8937 (on Solaris 10/Sparc)
|
# workaround for JBEAP-8937 (on Solaris 10/Sparc)
|
||||||
if [ -d "${wdir}" ]; then
|
if [ -d "${wdir}" ]; then
|
||||||
wdir=$(
|
wdir=`cd "$wdir/.."; pwd`
|
||||||
cd "$wdir/.."
|
|
||||||
pwd
|
|
||||||
)
|
|
||||||
fi
|
fi
|
||||||
# end of workaround
|
# end of workaround
|
||||||
done
|
done
|
||||||
@@ -203,9 +194,9 @@ concat_lines() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
BASE_DIR=$(find_maven_basedir "$(pwd)")
|
BASE_DIR=`find_maven_basedir "$(pwd)"`
|
||||||
if [ -z "$BASE_DIR" ]; then
|
if [ -z "$BASE_DIR" ]; then
|
||||||
exit 1
|
exit 1;
|
||||||
fi
|
fi
|
||||||
|
|
||||||
##########################################################################################
|
##########################################################################################
|
||||||
@@ -221,15 +212,12 @@ else
|
|||||||
echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..."
|
echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..."
|
||||||
fi
|
fi
|
||||||
if [ -n "$MVNW_REPOURL" ]; then
|
if [ -n "$MVNW_REPOURL" ]; then
|
||||||
jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.5/maven-wrapper-0.5.5.jar"
|
jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||||
else
|
else
|
||||||
jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.5/maven-wrapper-0.5.5.jar"
|
jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||||
fi
|
fi
|
||||||
while IFS="=" read key value; do
|
while IFS="=" read key value; do
|
||||||
case "$key" in wrapperUrl)
|
case "$key" in (wrapperUrl) jarUrl="$value"; break ;;
|
||||||
jarUrl="$value"
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
esac
|
esac
|
||||||
done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties"
|
done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties"
|
||||||
if [ "$MVNW_VERBOSE" = true ]; then
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
@@ -237,7 +225,7 @@ else
|
|||||||
fi
|
fi
|
||||||
wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar"
|
wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar"
|
||||||
if $cygwin; then
|
if $cygwin; then
|
||||||
wrapperJarPath=$(cygpath --path --windows "$wrapperJarPath")
|
wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"`
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if command -v wget > /dev/null; then
|
if command -v wget > /dev/null; then
|
||||||
@@ -266,7 +254,7 @@ else
|
|||||||
javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java"
|
javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java"
|
||||||
# For Cygwin, switch paths to Windows format before running javac
|
# For Cygwin, switch paths to Windows format before running javac
|
||||||
if $cygwin; then
|
if $cygwin; then
|
||||||
javaClass=$(cygpath --path --windows "$javaClass")
|
javaClass=`cygpath --path --windows "$javaClass"`
|
||||||
fi
|
fi
|
||||||
if [ -e "$javaClass" ]; then
|
if [ -e "$javaClass" ]; then
|
||||||
if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||||
@@ -299,13 +287,13 @@ MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS"
|
|||||||
# For Cygwin, switch paths to Windows format before running java
|
# For Cygwin, switch paths to Windows format before running java
|
||||||
if $cygwin; then
|
if $cygwin; then
|
||||||
[ -n "$M2_HOME" ] &&
|
[ -n "$M2_HOME" ] &&
|
||||||
M2_HOME=$(cygpath --path --windows "$M2_HOME")
|
M2_HOME=`cygpath --path --windows "$M2_HOME"`
|
||||||
[ -n "$JAVA_HOME" ] &&
|
[ -n "$JAVA_HOME" ] &&
|
||||||
JAVA_HOME=$(cygpath --path --windows "$JAVA_HOME")
|
JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"`
|
||||||
[ -n "$CLASSPATH" ] &&
|
[ -n "$CLASSPATH" ] &&
|
||||||
CLASSPATH=$(cygpath --path --windows "$CLASSPATH")
|
CLASSPATH=`cygpath --path --windows "$CLASSPATH"`
|
||||||
[ -n "$MAVEN_PROJECTBASEDIR" ] &&
|
[ -n "$MAVEN_PROJECTBASEDIR" ] &&
|
||||||
MAVEN_PROJECTBASEDIR=$(cygpath --path --windows "$MAVEN_PROJECTBASEDIR")
|
MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"`
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Provide a "standardized" way to retrieve the CLI args that will
|
# Provide a "standardized" way to retrieve the CLI args that will
|
||||||
|
|||||||
8
mvnw.cmd
vendored
8
mvnw.cmd
vendored
@@ -7,7 +7,7 @@
|
|||||||
@REM "License"); you may not use this file except in compliance
|
@REM "License"); you may not use this file except in compliance
|
||||||
@REM with the License. You may obtain a copy of the License at
|
@REM with the License. You may obtain a copy of the License at
|
||||||
@REM
|
@REM
|
||||||
@REM https://www.apache.org/licenses/LICENSE-2.0
|
@REM http://www.apache.org/licenses/LICENSE-2.0
|
||||||
@REM
|
@REM
|
||||||
@REM Unless required by applicable law or agreed to in writing,
|
@REM Unless required by applicable law or agreed to in writing,
|
||||||
@REM software distributed under the License is distributed on an
|
@REM software distributed under the License is distributed on an
|
||||||
@@ -18,7 +18,7 @@
|
|||||||
@REM ----------------------------------------------------------------------------
|
@REM ----------------------------------------------------------------------------
|
||||||
|
|
||||||
@REM ----------------------------------------------------------------------------
|
@REM ----------------------------------------------------------------------------
|
||||||
@REM Maven2 Start Up Batch script
|
@REM Maven Start Up Batch script
|
||||||
@REM
|
@REM
|
||||||
@REM Required ENV vars:
|
@REM Required ENV vars:
|
||||||
@REM JAVA_HOME - location of a JDK home dir
|
@REM JAVA_HOME - location of a JDK home dir
|
||||||
@@ -120,7 +120,7 @@ SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe"
|
|||||||
set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
|
set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
|
||||||
set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||||
|
|
||||||
set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.5/maven-wrapper-0.5.5.jar"
|
set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||||
|
|
||||||
FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO (
|
FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO (
|
||||||
IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
|
IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
|
||||||
@@ -134,7 +134,7 @@ if exist %WRAPPER_JAR% (
|
|||||||
)
|
)
|
||||||
) else (
|
) else (
|
||||||
if not "%MVNW_REPOURL%" == "" (
|
if not "%MVNW_REPOURL%" == "" (
|
||||||
SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.5/maven-wrapper-0.5.5.jar"
|
SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||||
)
|
)
|
||||||
if "%MVNW_VERBOSE%" == "true" (
|
if "%MVNW_VERBOSE%" == "true" (
|
||||||
echo Couldn't find %WRAPPER_JAR%, downloading it ...
|
echo Couldn't find %WRAPPER_JAR%, downloading it ...
|
||||||
|
|||||||
53
pom.xml
53
pom.xml
@@ -148,6 +148,41 @@
|
|||||||
<artifactId>jjwt</artifactId>
|
<artifactId>jjwt</artifactId>
|
||||||
<version>0.9.1</version>
|
<version>0.9.1</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
|
<!-- OkHttp -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.squareup.okhttp3</groupId>
|
||||||
|
<artifactId>okhttp</artifactId>
|
||||||
|
<version>4.8.0</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-stdlib</artifactId>
|
||||||
|
<version>1.3.72</version>
|
||||||
|
<scope>compile</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>net.sourceforge.htmlunit</groupId>
|
||||||
|
<artifactId>htmlunit</artifactId>
|
||||||
|
<version>2.42.0</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.h2database</groupId>
|
||||||
|
<artifactId>h2</artifactId>
|
||||||
|
<version>1.4.200</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.github.kstyrc</groupId>
|
||||||
|
<artifactId>embedded-redis</artifactId>
|
||||||
|
<version>0.6</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
@@ -157,21 +192,15 @@
|
|||||||
<artifactId>spring-boot-maven-plugin</artifactId>
|
<artifactId>spring-boot-maven-plugin</artifactId>
|
||||||
</plugin>
|
</plugin>
|
||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.jacoco</groupId>
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
<artifactId>jacoco-maven-plugin</artifactId>
|
<artifactId>kotlin-maven-plugin</artifactId>
|
||||||
<version>0.7.9</version>
|
<version>1.3.72</version>
|
||||||
<executions>
|
<executions>
|
||||||
<execution>
|
<execution>
|
||||||
<id>pre-unit-test</id>
|
<id>compile</id>
|
||||||
|
<phase>process-sources</phase>
|
||||||
<goals>
|
<goals>
|
||||||
<goal>prepare-agent</goal>
|
<goal>compile</goal>
|
||||||
</goals>
|
|
||||||
</execution>
|
|
||||||
<execution>
|
|
||||||
<id>post-unit-test</id>
|
|
||||||
<phase>test</phase>
|
|
||||||
<goals>
|
|
||||||
<goal>report</goal>
|
|
||||||
</goals>
|
</goals>
|
||||||
</execution>
|
</execution>
|
||||||
</executions>
|
</executions>
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public class CorsConfig {
|
|||||||
config.addAllowedOrigin("https://celess.cn");
|
config.addAllowedOrigin("https://celess.cn");
|
||||||
config.addAllowedOrigin("https://www.celess.cn");
|
config.addAllowedOrigin("https://www.celess.cn");
|
||||||
// 本地调试时的跨域
|
// 本地调试时的跨域
|
||||||
if ("dev".equals(activeModel)) {
|
if (!"prod".equals(activeModel)) {
|
||||||
config.addAllowedOrigin("http://localhost:4200");
|
config.addAllowedOrigin("http://localhost:4200");
|
||||||
config.addAllowedOrigin("http://127.0.0.1:4200");
|
config.addAllowedOrigin("http://127.0.0.1:4200");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public class DruidConfig {
|
|||||||
@Bean
|
@Bean
|
||||||
public DruidDataSource druidDataSource() {
|
public DruidDataSource druidDataSource() {
|
||||||
DruidDataSource dataSource = new DruidDataSource();
|
DruidDataSource dataSource = new DruidDataSource();
|
||||||
dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
|
dataSource.setDriverClassName(driverClassName);
|
||||||
// 数据库基本信息
|
// 数据库基本信息
|
||||||
dataSource.setUrl(dbUrl);
|
dataSource.setUrl(dbUrl);
|
||||||
dataSource.setUsername(username);
|
dataSource.setUsername(username);
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public class SwaggerConfig {
|
|||||||
@Bean
|
@Bean
|
||||||
public Docket createRestApi() {
|
public Docket createRestApi() {
|
||||||
return new Docket(DocumentationType.SWAGGER_2)
|
return new Docket(DocumentationType.SWAGGER_2)
|
||||||
.enable("dev".equals(environment))
|
.enable(!"prod".equals(environment))
|
||||||
.apiInfo(apiInfo())
|
.apiInfo(apiInfo())
|
||||||
.select()
|
.select()
|
||||||
.apis(RequestHandlerSelectors.basePackage("cn.celess.blog"))
|
.apis(RequestHandlerSelectors.basePackage("cn.celess.blog"))
|
||||||
|
|||||||
@@ -56,10 +56,13 @@ public class AuthenticationFilter implements HandlerInterceptor {
|
|||||||
return writeResponse(ResponseEnum.LOGIN_EXPIRED, response, request);
|
return writeResponse(ResponseEnum.LOGIN_EXPIRED, response, request);
|
||||||
}
|
}
|
||||||
String email = jwtUtil.getUsernameFromToken(jwtStr);
|
String email = jwtUtil.getUsernameFromToken(jwtStr);
|
||||||
if (!redisUtil.hasKey(email + "-login") || jwtUtil.isTokenExpired(jwtStr)) {
|
if (jwtUtil.isTokenExpired(jwtStr)) {
|
||||||
// 登陆过期
|
// 登陆过期
|
||||||
return writeResponse(ResponseEnum.LOGIN_EXPIRED, response, request);
|
return writeResponse(ResponseEnum.LOGIN_EXPIRED, response, request);
|
||||||
}
|
}
|
||||||
|
if (!redisUtil.hasKey(email + "-login")) {
|
||||||
|
return writeResponse(ResponseEnum.LOGOUT, response, request);
|
||||||
|
}
|
||||||
String role = userService.getUserRoleByEmail(email);
|
String role = userService.getUserRoleByEmail(email);
|
||||||
if (role.equals(ROLE_USER) || role.equals(ROLE_ADMIN)) {
|
if (role.equals(ROLE_USER) || role.equals(ROLE_ADMIN)) {
|
||||||
// 更新token
|
// 更新token
|
||||||
|
|||||||
@@ -135,8 +135,6 @@ public class CommonController {
|
|||||||
* FUCK !!!
|
* FUCK !!!
|
||||||
*
|
*
|
||||||
* @param file 文件
|
* @param file 文件
|
||||||
* @return
|
|
||||||
* @throws IOException
|
|
||||||
*/
|
*/
|
||||||
@PostMapping("/imgUpload")
|
@PostMapping("/imgUpload")
|
||||||
public void upload(HttpServletRequest request, HttpServletResponse response, @RequestParam("editormd-image-file") MultipartFile file) throws IOException {
|
public void upload(HttpServletRequest request, HttpServletResponse response, @RequestParam("editormd-image-file") MultipartFile file) throws IOException {
|
||||||
@@ -159,6 +157,7 @@ public class CommonController {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
String fileName = file.getOriginalFilename();
|
String fileName = file.getOriginalFilename();
|
||||||
|
assert fileName != null;
|
||||||
String mime = fileName.substring(fileName.lastIndexOf("."));
|
String mime = fileName.substring(fileName.lastIndexOf("."));
|
||||||
if (".png".equals(mime.toLowerCase()) || ".jpg".equals(mime.toLowerCase()) ||
|
if (".png".equals(mime.toLowerCase()) || ".jpg".equals(mime.toLowerCase()) ||
|
||||||
".jpeg".equals(mime.toLowerCase()) || ".bmp".equals(mime.toLowerCase())) {
|
".jpeg".equals(mime.toLowerCase()) || ".bmp".equals(mime.toLowerCase())) {
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package cn.celess.blog.controller;
|
|||||||
import cn.celess.blog.enmu.ResponseEnum;
|
import cn.celess.blog.enmu.ResponseEnum;
|
||||||
import cn.celess.blog.entity.PartnerSite;
|
import cn.celess.blog.entity.PartnerSite;
|
||||||
import cn.celess.blog.entity.Response;
|
import cn.celess.blog.entity.Response;
|
||||||
|
import cn.celess.blog.entity.request.LinkApplyReq;
|
||||||
import cn.celess.blog.entity.request.LinkReq;
|
import cn.celess.blog.entity.request.LinkReq;
|
||||||
import cn.celess.blog.exception.MyException;
|
import cn.celess.blog.exception.MyException;
|
||||||
import cn.celess.blog.service.MailService;
|
import cn.celess.blog.service.MailService;
|
||||||
@@ -69,29 +70,12 @@ public class LinksController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@PostMapping("/apply")
|
@PostMapping("/apply")
|
||||||
public Response apply(@RequestParam("name") String name,
|
public Response apply(@RequestBody() LinkApplyReq linkApplyReq) {
|
||||||
@RequestParam("url") String url) {
|
return Response.success(partnerSiteService.apply(linkApplyReq));
|
||||||
// TODO :: 弃用发送邮件的方式。
|
|
||||||
if (name == null || name.replaceAll(" ", "").isEmpty()) {
|
|
||||||
return Response.response(ResponseEnum.PARAMETERS_ERROR, null);
|
|
||||||
}
|
}
|
||||||
if (!RegexUtil.urlMatch(url)) {
|
|
||||||
return Response.response(ResponseEnum.PARAMETERS_URL_ERROR, null);
|
@PostMapping("/reapply")
|
||||||
}
|
public Response reapply(@RequestParam("key") String key) {
|
||||||
String applyTimeStr = redisUtil.get(request.getRemoteAddr() + "-Apply");
|
return Response.success(partnerSiteService.reapply(key));
|
||||||
int applyTime = 0;
|
|
||||||
if (applyTimeStr != null) {
|
|
||||||
applyTime = Integer.parseInt(applyTimeStr);
|
|
||||||
}
|
|
||||||
if (applyTime == 10) {
|
|
||||||
throw new MyException(ResponseEnum.FAILURE.getCode(), "申请次数已达10次,请2小时后重试");
|
|
||||||
}
|
|
||||||
SimpleMailMessage message = new SimpleMailMessage();
|
|
||||||
message.setSubject("友链申请:" + name);
|
|
||||||
message.setTo("a@celess.cn");
|
|
||||||
message.setText("name:" + name + "\nurl:" + url + "\n" + DateFormatUtil.getNow());
|
|
||||||
Boolean send = mailService.send(message);
|
|
||||||
redisUtil.setEx(request.getRemoteAddr() + "-Apply", applyTime + 1 + "", 2, TimeUnit.HOURS);
|
|
||||||
return send ? Response.success("") : Response.failure("");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ public enum ResponseEnum {
|
|||||||
PWD_SAME(3601, "新密码与原密码相同"),
|
PWD_SAME(3601, "新密码与原密码相同"),
|
||||||
PWD_NOT_SAME(3602, "新密码与原密码不相同"),
|
PWD_NOT_SAME(3602, "新密码与原密码不相同"),
|
||||||
LOGIN_EXPIRED(3700, "登陆过期"),
|
LOGIN_EXPIRED(3700, "登陆过期"),
|
||||||
|
LOGOUT(3710, "账户已注销"),
|
||||||
PWD_WRONG(3800, "密码不正确"),
|
PWD_WRONG(3800, "密码不正确"),
|
||||||
|
|
||||||
JWT_EXPIRED(3810, "Token过期"),
|
JWT_EXPIRED(3810, "Token过期"),
|
||||||
@@ -57,6 +58,9 @@ public enum ResponseEnum {
|
|||||||
DATA_HAS_EXIST(7020, "数据已存在"),
|
DATA_HAS_EXIST(7020, "数据已存在"),
|
||||||
|
|
||||||
//其他
|
//其他
|
||||||
|
APPLY_LINK_NO_ADD_THIS_SITE(7200, "暂未在您的网站中抓取到本站链接"),
|
||||||
|
DATA_EXPIRED(7300, "数据过期"),
|
||||||
|
CANNOT_GET_DATA(7400, "暂无法获取到数据"),
|
||||||
|
|
||||||
//提交更新之前,没有获取数据/,
|
//提交更新之前,没有获取数据/,
|
||||||
DID_NOT_GET_THE_DATA(8020, "非法访问"),
|
DID_NOT_GET_THE_DATA(8020, "非法访问"),
|
||||||
|
|||||||
@@ -25,6 +25,10 @@ public class PartnerSite {
|
|||||||
|
|
||||||
private Boolean delete = false;
|
private Boolean delete = false;
|
||||||
|
|
||||||
|
private String email;
|
||||||
|
|
||||||
|
private Boolean notification = true;
|
||||||
|
|
||||||
public PartnerSite() {
|
public PartnerSite() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,17 @@
|
|||||||
|
package cn.celess.blog.entity.request;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author : xiaohai
|
||||||
|
* @date : 2020/07/31 20:50
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class LinkApplyReq {
|
||||||
|
private String name;
|
||||||
|
private String email;
|
||||||
|
private String url;
|
||||||
|
private String linkUrl;
|
||||||
|
private String desc;
|
||||||
|
private String iconPath;
|
||||||
|
}
|
||||||
@@ -39,8 +39,9 @@ public class ExceptionHandle {
|
|||||||
public Response handle(Exception e) {
|
public Response handle(Exception e) {
|
||||||
//自定义错误
|
//自定义错误
|
||||||
if (e instanceof MyException) {
|
if (e instanceof MyException) {
|
||||||
logger.debug("返回了自定义的exception,[code={},msg={}]", ((MyException) e).getCode(), e.getMessage());
|
MyException exception = (MyException) e;
|
||||||
return new Response(((MyException) e).getCode(), e.getMessage(), null);
|
logger.debug("返回了自定义的exception,[code={},msg={},result={}]", exception.getCode(), e.getMessage(), exception.getResult());
|
||||||
|
return new Response(exception.getCode(), e.getMessage(), exception.getResult());
|
||||||
}
|
}
|
||||||
//请求路径不支持该方法
|
//请求路径不支持该方法
|
||||||
if (e instanceof HttpRequestMethodNotSupportedException) {
|
if (e instanceof HttpRequestMethodNotSupportedException) {
|
||||||
|
|||||||
@@ -1,13 +1,16 @@
|
|||||||
package cn.celess.blog.exception;
|
package cn.celess.blog.exception;
|
||||||
|
|
||||||
import cn.celess.blog.enmu.ResponseEnum;
|
import cn.celess.blog.enmu.ResponseEnum;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author : xiaohai
|
* @author : xiaohai
|
||||||
* @date : 2019/03/28 16:56
|
* @date : 2019/03/28 16:56
|
||||||
*/
|
*/
|
||||||
|
@Data
|
||||||
public class MyException extends RuntimeException {
|
public class MyException extends RuntimeException {
|
||||||
private int code;
|
private int code;
|
||||||
|
private Object result;
|
||||||
|
|
||||||
public MyException(int code, String msg) {
|
public MyException(int code, String msg) {
|
||||||
super(msg);
|
super(msg);
|
||||||
@@ -24,11 +27,9 @@ public class MyException extends RuntimeException {
|
|||||||
this.code = e.getCode();
|
this.code = e.getCode();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getCode() {
|
public MyException(ResponseEnum e, String msg, Object result) {
|
||||||
return code;
|
super(e.getMsg());
|
||||||
}
|
this.code = e.getCode();
|
||||||
|
this.result = result;
|
||||||
public void setCode(int code) {
|
|
||||||
this.code = code;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package cn.celess.blog.service;
|
|||||||
|
|
||||||
import cn.celess.blog.entity.PartnerSite;
|
import cn.celess.blog.entity.PartnerSite;
|
||||||
import cn.celess.blog.entity.model.PageData;
|
import cn.celess.blog.entity.model.PageData;
|
||||||
|
import cn.celess.blog.entity.request.LinkApplyReq;
|
||||||
import cn.celess.blog.entity.request.LinkReq;
|
import cn.celess.blog.entity.request.LinkReq;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@@ -53,4 +54,19 @@ public interface PartnerSiteService {
|
|||||||
*/
|
*/
|
||||||
List<PartnerSite> findAll();
|
List<PartnerSite> findAll();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 申请友链
|
||||||
|
*
|
||||||
|
* @param linkApplyReq linkApplyReq
|
||||||
|
* @return linkApplyReq
|
||||||
|
*/
|
||||||
|
PartnerSite apply(LinkApplyReq linkApplyReq);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 重写申请友链
|
||||||
|
*
|
||||||
|
* @param key key
|
||||||
|
* @return msg
|
||||||
|
*/
|
||||||
|
String reapply(String key);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ public class CommentServiceImpl implements CommentService {
|
|||||||
Comment comment = new Comment();
|
Comment comment = new Comment();
|
||||||
comment.setFromUser(user);
|
comment.setFromUser(user);
|
||||||
User userTo = new User();
|
User userTo = new User();
|
||||||
userTo.setId(-1L);
|
userTo.setId(null);
|
||||||
if (reqBody.getToUserId() != -1) {
|
if (reqBody.getToUserId() != -1) {
|
||||||
userTo = userMapper.findById(reqBody.getToUserId());
|
userTo = userMapper.findById(reqBody.getToUserId());
|
||||||
comment.setToUser(userTo);
|
comment.setToUser(userTo);
|
||||||
|
|||||||
@@ -3,18 +3,30 @@ package cn.celess.blog.service.serviceimpl;
|
|||||||
import cn.celess.blog.enmu.ResponseEnum;
|
import cn.celess.blog.enmu.ResponseEnum;
|
||||||
import cn.celess.blog.entity.PartnerSite;
|
import cn.celess.blog.entity.PartnerSite;
|
||||||
import cn.celess.blog.entity.model.PageData;
|
import cn.celess.blog.entity.model.PageData;
|
||||||
|
import cn.celess.blog.entity.request.LinkApplyReq;
|
||||||
import cn.celess.blog.entity.request.LinkReq;
|
import cn.celess.blog.entity.request.LinkReq;
|
||||||
import cn.celess.blog.exception.MyException;
|
import cn.celess.blog.exception.MyException;
|
||||||
import cn.celess.blog.mapper.PartnerMapper;
|
import cn.celess.blog.mapper.PartnerMapper;
|
||||||
|
import cn.celess.blog.service.MailService;
|
||||||
import cn.celess.blog.service.PartnerSiteService;
|
import cn.celess.blog.service.PartnerSiteService;
|
||||||
|
import cn.celess.blog.util.HttpUtil;
|
||||||
|
import cn.celess.blog.util.JwtUtil;
|
||||||
|
import cn.celess.blog.util.RedisUtil;
|
||||||
import cn.celess.blog.util.RegexUtil;
|
import cn.celess.blog.util.RegexUtil;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import com.github.pagehelper.PageHelper;
|
import com.github.pagehelper.PageHelper;
|
||||||
import com.github.pagehelper.PageInfo;
|
import com.github.pagehelper.PageInfo;
|
||||||
|
import lombok.SneakyThrows;
|
||||||
|
import org.apache.commons.lang.StringUtils;
|
||||||
import org.springframework.beans.BeanUtils;
|
import org.springframework.beans.BeanUtils;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.mail.SimpleMailMessage;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author : xiaohai
|
* @author : xiaohai
|
||||||
@@ -24,6 +36,13 @@ import java.util.List;
|
|||||||
public class PartnerSiteServiceImpl implements PartnerSiteService {
|
public class PartnerSiteServiceImpl implements PartnerSiteService {
|
||||||
@Autowired
|
@Autowired
|
||||||
PartnerMapper partnerMapper;
|
PartnerMapper partnerMapper;
|
||||||
|
@Autowired
|
||||||
|
MailService mailService;
|
||||||
|
@Autowired
|
||||||
|
RedisUtil redisUtil;
|
||||||
|
private static final String SITE_NAME = "小海博客";
|
||||||
|
private static final String SITE_URL = "celess.cn";
|
||||||
|
private static final String SITE_EMAIL = "a@celess.cn";
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PartnerSite create(LinkReq reqBody) {
|
public PartnerSite create(LinkReq reqBody) {
|
||||||
@@ -86,10 +105,21 @@ public class PartnerSiteServiceImpl implements PartnerSiteService {
|
|||||||
if (!reqBody.getUrl().contains("http://") && !reqBody.getUrl().contains("https://")) {
|
if (!reqBody.getUrl().contains("http://") && !reqBody.getUrl().contains("https://")) {
|
||||||
reqBody.setUrl("http://" + reqBody.getUrl());
|
reqBody.setUrl("http://" + reqBody.getUrl());
|
||||||
}
|
}
|
||||||
|
if (reqBody.isOpen() != partnerSite.getOpen() && !partnerSite.getNotification() && !StringUtils.isEmpty(partnerSite.getEmail())) {
|
||||||
|
SimpleMailMessage smm = new SimpleMailMessage();
|
||||||
|
smm.setTo(partnerSite.getEmail());
|
||||||
|
smm.setText("您的友链申请,已通过");
|
||||||
|
smm.setSubject("友链申请通过");
|
||||||
|
smm.setSentDate(new Date());
|
||||||
|
mailService.send(smm);
|
||||||
|
partnerSite.setNotification(true);
|
||||||
|
}
|
||||||
|
BeanUtils.copyProperties(reqBody, partnerSite);
|
||||||
|
partnerMapper.update(partnerSite);
|
||||||
partnerSite.setName(reqBody.getName());
|
partnerSite.setName(reqBody.getName());
|
||||||
partnerSite.setUrl(reqBody.getUrl());
|
partnerSite.setUrl(reqBody.getUrl());
|
||||||
partnerSite.setOpen(reqBody.isOpen());
|
partnerSite.setOpen(reqBody.isOpen());
|
||||||
partnerMapper.update(partnerSite);
|
|
||||||
return partnerSite;
|
return partnerSite;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -108,5 +138,92 @@ public class PartnerSiteServiceImpl implements PartnerSiteService {
|
|||||||
return all;
|
return all;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SneakyThrows
|
||||||
|
@Override
|
||||||
|
public PartnerSite apply(LinkApplyReq linkApplyReq) {
|
||||||
|
// 空值字段
|
||||||
|
if (StringUtils.isEmpty(linkApplyReq.getName())
|
||||||
|
|| StringUtils.isEmpty(linkApplyReq.getUrl())
|
||||||
|
|| StringUtils.isEmpty(linkApplyReq.getEmail())
|
||||||
|
|| StringUtils.isEmpty(linkApplyReq.getLinkUrl())) {
|
||||||
|
throw new MyException(ResponseEnum.PARAMETERS_ERROR);
|
||||||
|
}
|
||||||
|
// 链接不合法
|
||||||
|
if (!RegexUtil.emailMatch(linkApplyReq.getEmail())) {
|
||||||
|
throw new MyException(ResponseEnum.PARAMETERS_EMAIL_ERROR);
|
||||||
|
}
|
||||||
|
if (!RegexUtil.urlMatch(linkApplyReq.getLinkUrl()) || !RegexUtil.urlMatch(linkApplyReq.getUrl())) {
|
||||||
|
throw new MyException(ResponseEnum.PARAMETERS_URL_ERROR);
|
||||||
|
}
|
||||||
|
if (!StringUtils.isEmpty(linkApplyReq.getIconPath()) && !RegexUtil.urlMatch(linkApplyReq.getIconPath())) {
|
||||||
|
throw new MyException(ResponseEnum.PARAMETERS_URL_ERROR);
|
||||||
|
}
|
||||||
|
// 非强制字段 设置空
|
||||||
|
if (StringUtils.isEmpty(linkApplyReq.getIconPath())) {
|
||||||
|
linkApplyReq.setIconPath("");
|
||||||
|
}
|
||||||
|
// 抓取页面
|
||||||
|
String resp = HttpUtil.getAfterRendering(linkApplyReq.getLinkUrl());
|
||||||
|
if (resp == null) {
|
||||||
|
throw new MyException(ResponseEnum.CANNOT_GET_DATA);
|
||||||
|
}
|
||||||
|
PartnerSite ps = new PartnerSite();
|
||||||
|
if (resp.contains(SITE_URL)) {
|
||||||
|
//包含站点
|
||||||
|
BeanUtils.copyProperties(linkApplyReq, ps);
|
||||||
|
ps.setNotification(false);
|
||||||
|
ps.setOpen(false);
|
||||||
|
boolean exists = partnerMapper.existsByUrl(linkApplyReq.getUrl());
|
||||||
|
if (!exists) {
|
||||||
|
partnerMapper.insert(ps);
|
||||||
|
} else {
|
||||||
|
ps.setId(partnerMapper.findByUrl(linkApplyReq.getUrl()).getId());
|
||||||
|
}
|
||||||
|
SimpleMailMessage smm = new SimpleMailMessage();
|
||||||
|
smm.setSubject("友链申请");
|
||||||
|
smm.setText("有一条友链申请" + (exists ? ",已存在的友链链接" : "") + ",[\n" + linkApplyReq.toString() + "\n]");
|
||||||
|
smm.setTo(SITE_EMAIL);
|
||||||
|
smm.setSentDate(new Date());
|
||||||
|
mailService.send(smm);
|
||||||
|
} else {
|
||||||
|
// 不包含站点
|
||||||
|
String uuid;
|
||||||
|
ObjectMapper mapper = new ObjectMapper();
|
||||||
|
if (redisUtil.hasKey(linkApplyReq.getUrl())) {
|
||||||
|
uuid = redisUtil.get(linkApplyReq.getUrl());
|
||||||
|
if (!redisUtil.hasKey(uuid)) {
|
||||||
|
redisUtil.setEx(uuid, mapper.writeValueAsString(linkApplyReq), 10, TimeUnit.MINUTES);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
uuid = UUID.randomUUID().toString().replaceAll("-", "");
|
||||||
|
redisUtil.setEx(uuid, mapper.writeValueAsString(linkApplyReq), 10, TimeUnit.MINUTES);
|
||||||
|
redisUtil.setEx(linkApplyReq.getUrl(), uuid, 10, TimeUnit.MINUTES);
|
||||||
|
}
|
||||||
|
throw new MyException(ResponseEnum.APPLY_LINK_NO_ADD_THIS_SITE, null, uuid);
|
||||||
|
}
|
||||||
|
return ps;
|
||||||
|
}
|
||||||
|
|
||||||
|
@SneakyThrows
|
||||||
|
@Override
|
||||||
|
public String reapply(String key) {
|
||||||
|
if (!redisUtil.hasKey(key)) {
|
||||||
|
throw new MyException(ResponseEnum.DATA_EXPIRED);
|
||||||
|
}
|
||||||
|
String s = redisUtil.get(key);
|
||||||
|
ObjectMapper mapper = new ObjectMapper();
|
||||||
|
LinkApplyReq linkApplyReq = mapper.readValue(s, LinkApplyReq.class);
|
||||||
|
if (linkApplyReq == null) {
|
||||||
|
throw new MyException(ResponseEnum.DATA_NOT_EXIST);
|
||||||
|
}
|
||||||
|
SimpleMailMessage smm = new SimpleMailMessage();
|
||||||
|
smm.setSubject("友链申请");
|
||||||
|
smm.setText("有一条未抓取到信息的友链申请,[\n" + linkApplyReq.toString() + "\n]");
|
||||||
|
smm.setTo(SITE_EMAIL);
|
||||||
|
smm.setSentDate(new Date());
|
||||||
|
mailService.send(smm);
|
||||||
|
redisUtil.delete(key);
|
||||||
|
redisUtil.delete(linkApplyReq.getUrl());
|
||||||
|
return "success";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -81,6 +81,7 @@ public class WebUpdateInfoServiceImpl implements WebUpdateInfoService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
||||||
public Map<String, Object> getLastestUpdateTime() {
|
public Map<String, Object> getLastestUpdateTime() {
|
||||||
Map<String, Object> map = new HashMap<>();
|
Map<String, Object> map = new HashMap<>();
|
||||||
map.put("lastUpdateTime", DateFormatUtil.get(webUpdateInfoMapper.getLastestOne().getUpdateTime()));
|
map.put("lastUpdateTime", DateFormatUtil.get(webUpdateInfoMapper.getLastestOne().getUpdateTime()));
|
||||||
|
|||||||
@@ -1,12 +1,15 @@
|
|||||||
package cn.celess.blog.util;
|
package cn.celess.blog.util;
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
import com.gargoylesoftware.htmlunit.BrowserVersion;
|
||||||
|
import com.gargoylesoftware.htmlunit.NicelyResynchronizingAjaxController;
|
||||||
|
import com.gargoylesoftware.htmlunit.WebClient;
|
||||||
|
import com.gargoylesoftware.htmlunit.html.HtmlPage;
|
||||||
|
import okhttp3.OkHttpClient;
|
||||||
|
import okhttp3.Request;
|
||||||
|
import okhttp3.Response;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.util.Objects;
|
||||||
import java.io.InputStreamReader;
|
|
||||||
import java.net.HttpURLConnection;
|
|
||||||
import java.net.URL;
|
|
||||||
import java.nio.charset.StandardCharsets;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Author: 小海
|
* @Author: 小海
|
||||||
@@ -14,38 +17,34 @@ import java.nio.charset.StandardCharsets;
|
|||||||
* @Desc:
|
* @Desc:
|
||||||
*/
|
*/
|
||||||
public class HttpUtil {
|
public class HttpUtil {
|
||||||
|
private static final OkHttpClient CLIENT = new OkHttpClient();
|
||||||
|
|
||||||
public static String get(String urlStr) throws IOException {
|
public static String get(String urlStr) {
|
||||||
|
Request request = new Request.Builder()
|
||||||
|
.url(urlStr)
|
||||||
|
.get()
|
||||||
|
.build();
|
||||||
|
try (Response response = CLIENT.newCall(request).execute()) {
|
||||||
|
return Objects.requireNonNull(response.body()).string();
|
||||||
|
} catch (IOException e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
StringBuffer sb = new StringBuffer();
|
|
||||||
|
|
||||||
HttpURLConnection urlConnection = null;
|
public static String getAfterRendering(String url) {
|
||||||
try {
|
try (final WebClient webClient = new WebClient(BrowserVersion.CHROME)) {
|
||||||
URL url = new URL(urlStr);
|
webClient.getOptions().setCssEnabled(false);
|
||||||
|
webClient.getOptions().setJavaScriptEnabled(true);
|
||||||
//打开http
|
webClient.getOptions().setThrowExceptionOnFailingStatusCode(false);
|
||||||
urlConnection = (HttpURLConnection) url.openConnection();
|
webClient.getOptions().setThrowExceptionOnScriptError(false);
|
||||||
urlConnection.setDoInput(true);
|
webClient.getOptions().setDownloadImages(false);
|
||||||
urlConnection.setRequestMethod("GET");
|
webClient.getOptions().setActiveXNative(false);
|
||||||
urlConnection.connect();
|
webClient.setAjaxController(new NicelyResynchronizingAjaxController());
|
||||||
|
final HtmlPage page = webClient.getPage(url);
|
||||||
try (
|
return page.asXml();
|
||||||
InputStream inputStream = urlConnection.getInputStream();
|
} catch (IOException e) {
|
||||||
InputStreamReader inputStreamReader = new InputStreamReader(inputStream, StandardCharsets.UTF_8);
|
return null;
|
||||||
BufferedReader bufferedReader = new BufferedReader(inputStreamReader)
|
|
||||||
) {
|
|
||||||
//将bufferReader的值给放到buffer里
|
|
||||||
String str = null;
|
|
||||||
while ((str = bufferedReader.readLine()) != null) {
|
|
||||||
sb.append(str);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} finally {
|
|
||||||
//断开连接
|
|
||||||
if (urlConnection != null) {
|
|
||||||
urlConnection.disconnect();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return sb.toString();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,6 +26,12 @@ spring.datasource.url=
|
|||||||
spring.datasource.username=
|
spring.datasource.username=
|
||||||
spring.datasource.password=
|
spring.datasource.password=
|
||||||
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
|
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
|
||||||
|
spring.datasource.platform=mysql
|
||||||
|
# never / always / embedded
|
||||||
|
spring.datasource.initialization-mode=never
|
||||||
|
spring.datasource.sql-script-encoding=utf-8
|
||||||
|
spring.datasource.schema=classpath:sql/schema.sql
|
||||||
|
spring.datasource.data=classpath:sql/data.sql
|
||||||
|
|
||||||
|
|
||||||
################## mybatis ##################
|
################## mybatis ##################
|
||||||
|
|||||||
96
src/main/resources/application-test.properties
Normal file
96
src/main/resources/application-test.properties
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
server.port=8081
|
||||||
|
|
||||||
|
# 七牛的密钥配置
|
||||||
|
qiniu.accessKey=
|
||||||
|
qiniu.secretKey=
|
||||||
|
qiniu.bucket=
|
||||||
|
# sitemap 存放地址
|
||||||
|
sitemap.path=
|
||||||
|
# 生成JWT时候的密钥
|
||||||
|
jwt.secret=sdaniod213k123123ipoeqowekqwe
|
||||||
|
|
||||||
|
##spring.jpa.show-sql=false
|
||||||
|
##spring.jpa.hibernate.ddl-auto=update
|
||||||
|
|
||||||
|
mybatis.type-handlers-package=cn.celess.blog.mapper.typehandler
|
||||||
|
logging.level.cn.celess.blog.mapper=debug
|
||||||
|
# 上传单个文件的大小
|
||||||
|
spring.servlet.multipart.max-file-size=10MB
|
||||||
|
# 上传文件的总大小
|
||||||
|
spring.servlet.multipart.max-request-size=10MB
|
||||||
|
|
||||||
|
spring.jackson.default-property-inclusion=non_null
|
||||||
|
|
||||||
|
|
||||||
|
################# 数据库 ##################
|
||||||
|
spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
|
||||||
|
|
||||||
|
#h2
|
||||||
|
spring.datasource.driver-class-name=org.h2.Driver
|
||||||
|
spring.datasource.url=jdbc:h2:mem:testdb;mode=mysql;DB_CLOSE_DELAY=-1;DATABASE_TO_UPPER=false
|
||||||
|
spring.datasource.username=sa
|
||||||
|
spring.datasource.password=
|
||||||
|
|
||||||
|
|
||||||
|
spring.datasource.platform=h2
|
||||||
|
spring.datasource.sql-script-encoding=utf-8
|
||||||
|
spring.datasource.initialization-mode=ALWAYS
|
||||||
|
spring.datasource.schema=classpath:sql/schema_h2.sql
|
||||||
|
spring.datasource.data=classpath:sql/data.sql
|
||||||
|
|
||||||
|
|
||||||
|
################## mybatis ##################
|
||||||
|
mybatis.mapper-locations=classpath:mapper/*.xml
|
||||||
|
mybatis.type-aliases-package=cn.celess.blog.entity
|
||||||
|
|
||||||
|
|
||||||
|
pagehelper.helper-dialect=mysql
|
||||||
|
pagehelper.reasonable=true
|
||||||
|
pagehelper.support-methods-arguments=true
|
||||||
|
pagehelper.params=count=countSql
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#### 用于nginx的代理 获取真实ip
|
||||||
|
server.use-forward-headers = true
|
||||||
|
server.tomcat.remote-ip-header = X-Real-IP
|
||||||
|
server.tomcat.protocol-header = X-Forwarded-Proto
|
||||||
|
|
||||||
|
|
||||||
|
############### email ##############
|
||||||
|
spring.mail.host=smtp.163.com
|
||||||
|
spring.mail.username=
|
||||||
|
spring.mail.password=
|
||||||
|
spring.mail.properties.mail.smtp.auth=true
|
||||||
|
spring.mail.properties.mail.smtp.starttls.enable=true
|
||||||
|
spring.mail.properties.mail.smtp.starttls.required=true
|
||||||
|
spring.mail.default-encoding=UTF-8
|
||||||
|
spring.mail.port=465
|
||||||
|
spring.mail.properties.mail.smtp.socketFactory.port=465
|
||||||
|
spring.mail.properties.mail.smtp.socketFactory.class=javax.net.ssl.SSLSocketFactory
|
||||||
|
spring.mail.properties.mail.smtp.socketFactory.fallback=false
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
############### redis ##############
|
||||||
|
|
||||||
|
# REDIS (RedisProperties)
|
||||||
|
# Redis数据库索引(默认为0)
|
||||||
|
spring.redis.database=1
|
||||||
|
# Redis服务器地址
|
||||||
|
spring.redis.host=127.0.0.1
|
||||||
|
# Redis服务器连接端口 解决端口冲突 防止使用本地的redis
|
||||||
|
spring.redis.port=6380
|
||||||
|
# Redis服务器连接密码(默认为空)
|
||||||
|
spring.redis.password=
|
||||||
|
# 连接池最大连接数(使用负值表示没有限制)
|
||||||
|
spring.redis.jedis.pool.max-active=-1
|
||||||
|
# 连接池最大阻塞等待时间(使用负值表示没有限制)
|
||||||
|
spring.redis.jedis.pool.max-wait=-1
|
||||||
|
# 连接池中的最大空闲连接
|
||||||
|
spring.redis.jedis.pool.max-idle=8
|
||||||
|
# 连接池中的最小空闲连接
|
||||||
|
spring.redis.jedis.pool.min-idle=0
|
||||||
|
# 连接超时时间(毫秒)
|
||||||
|
spring.redis.timeout=5000
|
||||||
|
|
||||||
@@ -9,20 +9,26 @@
|
|||||||
<result column="l_icon_path" property="iconPath"/>
|
<result column="l_icon_path" property="iconPath"/>
|
||||||
<result column="l_desc" property="desc"/>
|
<result column="l_desc" property="desc"/>
|
||||||
<result column="is_delete" property="delete"/>
|
<result column="is_delete" property="delete"/>
|
||||||
|
<result column="l_email" property="email"/>
|
||||||
|
<result column="l_notification" property="notification"/>
|
||||||
</resultMap>
|
</resultMap>
|
||||||
|
|
||||||
<insert id="insert" parameterType="cn.celess.blog.entity.PartnerSite" useGeneratedKeys="true" keyProperty="id">
|
<insert id="insert" parameterType="cn.celess.blog.entity.PartnerSite" useGeneratedKeys="true" keyProperty="id">
|
||||||
insert into links (l_name, l_is_open, l_url, l_icon_path, l_desc, is_delete)
|
insert into links (l_name, l_is_open, l_url, l_icon_path, l_desc, l_email, l_notification, is_delete)
|
||||||
values (#{name}, #{open}, #{url}, #{iconPath}, #{desc}, false)
|
values (#{name}, #{open}, #{url}, #{iconPath}, #{desc}, #{email}, #{notification}, false)
|
||||||
</insert>
|
</insert>
|
||||||
|
|
||||||
<update id="update" parameterType="cn.celess.blog.entity.PartnerSite">
|
<update id="update" parameterType="cn.celess.blog.entity.PartnerSite">
|
||||||
update links set
|
update links
|
||||||
|
<trim prefix="set" suffixOverrides=",">
|
||||||
<if test="open!=null">l_is_open=#{open},</if>
|
<if test="open!=null">l_is_open=#{open},</if>
|
||||||
<if test="iconPath!=null">l_icon_path=#{iconPath},</if>
|
<if test="iconPath!=null">l_icon_path=#{iconPath},</if>
|
||||||
<if test="desc!=null">l_desc=#{desc},</if>
|
<if test="desc!=null">l_desc=#{desc},</if>
|
||||||
<if test="url!=null">l_url=#{url},</if>
|
<if test="url!=null">l_url=#{url},</if>
|
||||||
<if test="name!=null">l_name=#{name}</if>
|
<if test="name!=null">l_name=#{name},</if>
|
||||||
|
<if test="notification!=null">l_notification=#{notification},</if>
|
||||||
|
<if test="email!=null">l_email=#{email}</if>
|
||||||
|
</trim>
|
||||||
where l_id=#{id}
|
where l_id=#{id}
|
||||||
</update>
|
</update>
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -1,6 +1,14 @@
|
|||||||
CREATE DATABASE if not exists `blog`;
|
drop view if exists articleView;
|
||||||
|
drop view if exists commentView;
|
||||||
|
drop table if exists article_tag;
|
||||||
|
drop table if exists comment;
|
||||||
|
drop table if exists article;
|
||||||
|
drop table if exists user;
|
||||||
|
drop table if exists tag_category;
|
||||||
|
drop table if exists links;
|
||||||
|
drop table if exists visitor;
|
||||||
|
drop table if exists web_update;
|
||||||
|
|
||||||
USE blog;
|
|
||||||
|
|
||||||
CREATE TABLE `user`
|
CREATE TABLE `user`
|
||||||
(
|
(
|
||||||
@@ -66,7 +74,9 @@ CREATE TABLE `comment`
|
|||||||
`co_status` tinyint not null default 0 comment '评论的状态',
|
`co_status` tinyint not null default 0 comment '评论的状态',
|
||||||
`co_pid` bigint not null default -1 comment '评论/留言的父id',
|
`co_pid` bigint not null default -1 comment '评论/留言的父id',
|
||||||
`co_from_author_id` int not null comment '留言者id',
|
`co_from_author_id` int not null comment '留言者id',
|
||||||
`co_to_author_id` int default null comment '父评论的作者id'
|
`co_to_author_id` int default null comment '父评论的作者id',
|
||||||
|
foreign key (co_from_author_id) references user (u_id),
|
||||||
|
foreign key (co_to_author_id) references user (u_id)
|
||||||
) DEFAULT CHARSET = utf8mb4
|
) DEFAULT CHARSET = utf8mb4
|
||||||
COLLATE utf8mb4_general_ci,comment '评论/留言表';
|
COLLATE utf8mb4_general_ci,comment '评论/留言表';
|
||||||
|
|
||||||
@@ -75,10 +85,12 @@ CREATE TABLE `links`
|
|||||||
`l_id` bigint(20) primary key auto_increment,
|
`l_id` bigint(20) primary key auto_increment,
|
||||||
`l_name` varchar(255) COLLATE utf8mb4_unicode_ci not null comment '友站名称',
|
`l_name` varchar(255) COLLATE utf8mb4_unicode_ci not null comment '友站名称',
|
||||||
`l_is_open` boolean default true comment '是否公开',
|
`l_is_open` boolean default true comment '是否公开',
|
||||||
`l_url` varchar(255) not null comment '首页地址',
|
`l_url` varchar(255) unique not null comment '首页地址',
|
||||||
`l_icon_path` varchar(255) not null comment '友链的icon地址',
|
`l_icon_path` varchar(255) not null comment '友链的icon地址',
|
||||||
`l_desc` varchar(255) COLLATE utf8mb4_unicode_ci not null comment '友链的说明描述',
|
`l_desc` varchar(255) COLLATE utf8mb4_unicode_ci not null comment '友链的说明描述',
|
||||||
`is_delete` boolean not null default false comment '该数据是否被删除'
|
`is_delete` boolean not null default false comment '该数据是否被删除',
|
||||||
|
`l_email` varchar(255) comment '网站管理员的邮箱',
|
||||||
|
`l_notification` boolean default false comment '是否通知了'
|
||||||
) comment '友站表';
|
) comment '友站表';
|
||||||
|
|
||||||
CREATE TABLE `visitor`
|
CREATE TABLE `visitor`
|
||||||
193
src/main/resources/sql/schema_h2.sql
Normal file
193
src/main/resources/sql/schema_h2.sql
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
drop view if exists articleView;
|
||||||
|
drop view if exists commentView;
|
||||||
|
drop table if exists article_tag;
|
||||||
|
drop table if exists comment;
|
||||||
|
drop table if exists article;
|
||||||
|
drop table if exists user;
|
||||||
|
drop table if exists tag_category;
|
||||||
|
drop table if exists links;
|
||||||
|
drop table if exists visitor;
|
||||||
|
drop table if exists web_update;
|
||||||
|
|
||||||
|
-- 用户表
|
||||||
|
CREATE TABLE `user`
|
||||||
|
(
|
||||||
|
`u_id` int not null primary key auto_increment,
|
||||||
|
`u_email` varchar(50) not null,
|
||||||
|
`u_pwd` varchar(40) not null comment '密码',
|
||||||
|
`u_email_status` boolean default false comment '邮箱验证状态',
|
||||||
|
`u_avatar` varchar(255) default null comment '用户头像',
|
||||||
|
`u_desc` tinytext default null comment '用户的描述',
|
||||||
|
`u_recently_landed_time` datetime default null comment '最近的登录时间',
|
||||||
|
`u_display_name` varchar(30) default null comment '展示的昵称',
|
||||||
|
`u_role` varchar(40) not null default 'user' comment '权限组',
|
||||||
|
`status` tinyint(1) not null default 0 comment '账户状态',
|
||||||
|
unique key `uni_user_id` (`u_id`),
|
||||||
|
unique key `uni_user_email` (`u_email`)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 标签和分类表
|
||||||
|
CREATE TABLE `tag_category`
|
||||||
|
(
|
||||||
|
`t_id` bigint(20) primary key auto_increment,
|
||||||
|
`t_name` varchar(255) not null,
|
||||||
|
`is_category` boolean not null default true,
|
||||||
|
`is_delete` boolean not null default false comment '该数据是否被删除'
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 文章表
|
||||||
|
CREATE TABLE `article`
|
||||||
|
(
|
||||||
|
`a_id` bigint(20) primary key auto_increment,
|
||||||
|
`a_title` varchar(255) not null unique comment '文章标题',
|
||||||
|
`a_summary` varchar(255) not null comment '文章摘要',
|
||||||
|
`a_md_content` longtext not null comment '文章Markdown内容',
|
||||||
|
`a_url` tinytext default null comment '转载文章的原文链接',
|
||||||
|
`a_author_id` int not null comment '作者id',
|
||||||
|
`a_is_original` boolean default true comment '文章是否原创',
|
||||||
|
`a_reading_number` int default 0 comment '文章阅读数',
|
||||||
|
`a_like` int default 0 comment '文章点赞数',
|
||||||
|
`a_dislike` int default 0 comment '文章不喜欢数',
|
||||||
|
`a_category_id` bigint not null comment '文章分类id',
|
||||||
|
`a_publish_date` datetime default CURRENT_TIMESTAMP comment '文章发布时间',
|
||||||
|
`a_update_date` datetime default null comment '文章的更新时间',
|
||||||
|
`a_is_open` boolean default true comment '文章是否可见',
|
||||||
|
`is_delete` boolean not null default false comment '该数据是否被删除',
|
||||||
|
foreign key (a_category_id) references tag_category (t_id),
|
||||||
|
foreign key (a_author_id) references user (u_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 文章标签表
|
||||||
|
CREATE TABLE `article_tag`
|
||||||
|
(
|
||||||
|
`at_id` bigint(20) primary key auto_increment,
|
||||||
|
`a_id` bigint(20) not null comment '文章id',
|
||||||
|
`t_id` bigint not null comment 'tag/category 的id',
|
||||||
|
foreign key (a_id) references article (a_id),
|
||||||
|
foreign key (t_id) references tag_category (t_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 评论/留言表
|
||||||
|
CREATE TABLE `comment`
|
||||||
|
(
|
||||||
|
`co_id` bigint(20) primary key auto_increment,
|
||||||
|
`co_page_path` varchar(255) not null comment '评论/留言的页面',
|
||||||
|
`co_content` text not null comment '评论/留言内容',
|
||||||
|
`co_date` datetime not null comment '评论/留言的日期',
|
||||||
|
`co_status` tinyint not null default 0 comment '评论的状态',
|
||||||
|
`co_pid` bigint not null default -1 comment '评论/留言的父id',
|
||||||
|
`co_from_author_id` int not null comment '留言者id',
|
||||||
|
`co_to_author_id` int default null comment '父评论的作者id',
|
||||||
|
foreign key (co_from_author_id) references user (u_id),
|
||||||
|
foreign key (co_to_author_id) references user (u_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 友站表
|
||||||
|
CREATE TABLE `links`
|
||||||
|
(
|
||||||
|
`l_id` bigint(20) primary key auto_increment,
|
||||||
|
`l_name` varchar(255) not null comment '友站名称',
|
||||||
|
`l_is_open` boolean default true comment '是否公开',
|
||||||
|
`l_url` varchar(255) unique not null comment '首页地址',
|
||||||
|
`l_icon_path` varchar(255) not null comment '友链的icon地址',
|
||||||
|
`l_desc` varchar(255) not null comment '友链的说明描述',
|
||||||
|
`is_delete` boolean not null default false comment '该数据是否被删除',
|
||||||
|
`l_email` varchar(255) comment '网站管理员的邮箱',
|
||||||
|
`l_notification` boolean default false comment '是否通知了'
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 访客表
|
||||||
|
CREATE TABLE `visitor`
|
||||||
|
(
|
||||||
|
`v_id` bigint(20) primary key auto_increment,
|
||||||
|
`v_date` datetime not null comment '访问时间',
|
||||||
|
`v_ip` varchar(255) not null comment '访客ip',
|
||||||
|
`v_user_agent` text comment '访客ua',
|
||||||
|
`is_delete` boolean not null default false comment '该数据是否被删除'
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 更新内容表
|
||||||
|
CREATE TABLE `web_update`
|
||||||
|
(
|
||||||
|
`wu_id` int primary key auto_increment,
|
||||||
|
`wu_info` varchar(255) not null comment '更新内容',
|
||||||
|
`wu_time` datetime not null comment '更新时间',
|
||||||
|
`is_delete` boolean not null default false comment '该数据是否被删除'
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE VIEW articleView
|
||||||
|
(articleId, title, summary, mdContent, url, isOriginal, readingCount, likeCount, dislikeCount,
|
||||||
|
publishDate, updateDate, isOpen,
|
||||||
|
categoryId, categoryName, tagId, tagName,
|
||||||
|
authorId, userEmail, userAvatar, userDisplayName, isDelete)
|
||||||
|
as
|
||||||
|
select article.a_id as articleId,
|
||||||
|
article.a_title as title,
|
||||||
|
article.a_summary as summary,
|
||||||
|
article.a_md_content as mdContent,
|
||||||
|
article.a_url as url,
|
||||||
|
article.a_is_original as isOriginal,
|
||||||
|
article.a_reading_number as readingCount,
|
||||||
|
article.a_like as likeCount,
|
||||||
|
article.a_dislike as dislikeCount,
|
||||||
|
article.a_publish_date as publishDate,
|
||||||
|
article.a_update_date as updateDate,
|
||||||
|
article.a_is_open as isOpen,
|
||||||
|
category.t_id as categoryId,
|
||||||
|
category.t_name as categoryName,
|
||||||
|
tag.t_id as tagId,
|
||||||
|
tag.t_name as tagName,
|
||||||
|
user.u_id as authorId,
|
||||||
|
user.u_email as userEmail,
|
||||||
|
user.u_avatar as userAvatar,
|
||||||
|
user.u_display_name as userDisplayName,
|
||||||
|
article.is_delete as isDelete
|
||||||
|
from article,
|
||||||
|
tag_category as tag,
|
||||||
|
tag_category as category,
|
||||||
|
article_tag,
|
||||||
|
user
|
||||||
|
where article.a_id = article_tag.a_id
|
||||||
|
and article_tag.t_id = tag.t_id
|
||||||
|
and tag.is_category = false
|
||||||
|
and article.a_category_id = category.t_id
|
||||||
|
and category.is_category = true
|
||||||
|
and article.a_author_id = user.u_id;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE VIEW commentView
|
||||||
|
(commentId, pagePath, content, date, status, pid, toAuthorId, toAuthorEmail, toAuthorDisplayName,
|
||||||
|
toAuthorAvatar, fromAuthorId, fromAuthorEmail, fromAuthorDisplayName,
|
||||||
|
fromAuthorAvatar)
|
||||||
|
as
|
||||||
|
select cuT.co_id as commentId,
|
||||||
|
cuT.co_page_path as pagePath,
|
||||||
|
cuT.co_content as content,
|
||||||
|
cuT.co_date as date,
|
||||||
|
cuT.co_status as status,
|
||||||
|
cuT.co_pid as pid,
|
||||||
|
cuT.co_to_author_id as toAuthorId,
|
||||||
|
cuT.toEmail as toAuthorEmail,
|
||||||
|
cuT.toDisplayName as toAuthorDisplayName,
|
||||||
|
cuT.toAvatar as toAuthorAvatar,
|
||||||
|
userFrom.u_id as fromAuthorId,
|
||||||
|
userFrom.u_email as fromAuthorEmail,
|
||||||
|
userFrom.u_display_name as fromAuthorDisplayName,
|
||||||
|
userFrom.u_avatar as fromAuthorAvatar
|
||||||
|
from (select comment.co_id,
|
||||||
|
comment.co_page_path,
|
||||||
|
comment.co_content,
|
||||||
|
comment.co_date,
|
||||||
|
comment.co_status,
|
||||||
|
comment.co_pid,
|
||||||
|
comment.co_from_author_id,
|
||||||
|
comment.co_to_author_id,
|
||||||
|
userTo.u_email as toEmail,
|
||||||
|
userTo.u_display_name as toDisplayName,
|
||||||
|
userTo.u_avatar as toAvatar
|
||||||
|
from comment
|
||||||
|
left join user userTo on (comment.co_to_author_id = userTo.u_id)
|
||||||
|
) as cuT,
|
||||||
|
user as userFrom
|
||||||
|
where cuT.co_from_author_id = userFrom.u_id;
|
||||||
|
|
||||||
@@ -2,10 +2,16 @@ package cn.celess.blog;
|
|||||||
|
|
||||||
|
|
||||||
import cn.celess.blog.entity.Response;
|
import cn.celess.blog.entity.Response;
|
||||||
|
import cn.celess.blog.entity.model.QiniuResponse;
|
||||||
import cn.celess.blog.entity.model.UserModel;
|
import cn.celess.blog.entity.model.UserModel;
|
||||||
import cn.celess.blog.entity.request.LoginReq;
|
import cn.celess.blog.entity.request.LoginReq;
|
||||||
|
import cn.celess.blog.service.MailService;
|
||||||
|
import cn.celess.blog.service.QiniuService;
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
import com.fasterxml.jackson.core.type.TypeReference;
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import com.qiniu.storage.model.FileInfo;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
@@ -15,6 +21,7 @@ import org.slf4j.LoggerFactory;
|
|||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.boot.test.context.SpringBootTest;
|
import org.springframework.boot.test.context.SpringBootTest;
|
||||||
import org.springframework.http.MediaType;
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.mail.SimpleMailMessage;
|
||||||
import org.springframework.mock.web.MockHttpSession;
|
import org.springframework.mock.web.MockHttpSession;
|
||||||
import org.springframework.test.context.ActiveProfiles;
|
import org.springframework.test.context.ActiveProfiles;
|
||||||
import org.springframework.test.context.junit4.SpringRunner;
|
import org.springframework.test.context.junit4.SpringRunner;
|
||||||
@@ -28,7 +35,10 @@ import org.springframework.web.context.WebApplicationContext;
|
|||||||
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
import java.io.UnsupportedEncodingException;
|
import java.io.UnsupportedEncodingException;
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
import static cn.celess.blog.enmu.ResponseEnum.SUCCESS;
|
import static cn.celess.blog.enmu.ResponseEnum.SUCCESS;
|
||||||
@@ -54,8 +64,7 @@ public class BaseTest {
|
|||||||
protected MockMvc mockMvc;
|
protected MockMvc mockMvc;
|
||||||
protected final static String Code = "code";
|
protected final static String Code = "code";
|
||||||
protected final static String Result = "result";
|
protected final static String Result = "result";
|
||||||
protected final static String USERE_MAIL = "zh56462271@qq.com";
|
|
||||||
protected final static String ADMIN_EMAIL = "a@celess.cn";
|
|
||||||
/**
|
/**
|
||||||
* jackson 序列化/反序列化Json
|
* jackson 序列化/反序列化Json
|
||||||
*/
|
*/
|
||||||
@@ -66,6 +75,9 @@ public class BaseTest {
|
|||||||
};
|
};
|
||||||
protected static final TypeReference<?> OBJECT_TYPE = new TypeReference<Response<Object>>() {
|
protected static final TypeReference<?> OBJECT_TYPE = new TypeReference<Response<Object>>() {
|
||||||
};
|
};
|
||||||
|
protected static final TypeReference<?> MAP_OBJECT_TYPE = new TypeReference<Response<Map<String, Object>>>() {
|
||||||
|
};
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private WebApplicationContext wac;
|
private WebApplicationContext wac;
|
||||||
protected MockHttpSession session;
|
protected MockHttpSession session;
|
||||||
@@ -260,4 +272,74 @@ public class BaseTest {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改 mailService 的实现类
|
||||||
|
*
|
||||||
|
* @param service service 类
|
||||||
|
* @param mailFiledName service 中自动注入的mailService字段名
|
||||||
|
*/
|
||||||
|
protected void mockInjectInstance(Object service, String mailFiledName, Object impl) {
|
||||||
|
Field field;
|
||||||
|
try {
|
||||||
|
field = service.getClass().getDeclaredField(mailFiledName);
|
||||||
|
field.setAccessible(true);
|
||||||
|
field.set(service, impl);
|
||||||
|
} catch (NoSuchFieldException | IllegalAccessException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Slf4j
|
||||||
|
public static class TestMailServiceImpl implements MailService {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean AsyncSend(SimpleMailMessage message) {
|
||||||
|
log.debug("异步邮件请求,SimpleMailMessage:[{}]", getJson(message));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean send(SimpleMailMessage message) {
|
||||||
|
log.debug("邮件请求,SimpleMailMessage:[{}]", getJson(message));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 转json
|
||||||
|
*
|
||||||
|
* @param o
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
private String getJson(Object o) {
|
||||||
|
ObjectMapper mapper = new ObjectMapper();
|
||||||
|
try {
|
||||||
|
return mapper.writeValueAsString(o);
|
||||||
|
} catch (JsonProcessingException e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Slf4j
|
||||||
|
public static class TestQiNiuServiceImpl implements QiniuService {
|
||||||
|
@Override
|
||||||
|
public QiniuResponse uploadFile(InputStream is, String fileName) {
|
||||||
|
QiniuResponse response = new QiniuResponse();
|
||||||
|
log.debug("上传文件请求,[fileName:{}]", fileName);
|
||||||
|
|
||||||
|
response.key = "key";
|
||||||
|
response.bucket = "bucket";
|
||||||
|
response.hash = "hash";
|
||||||
|
response.fsize = 1;
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public FileInfo[] getFileList() {
|
||||||
|
log.debug("获取文件列表请求");
|
||||||
|
return new FileInfo[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
37
src/test/java/cn/celess/blog/RedisServerMock.java
Normal file
37
src/test/java/cn/celess/blog/RedisServerMock.java
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
package cn.celess.blog;
|
||||||
|
|
||||||
|
import redis.embedded.RedisServer;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
|
import javax.annotation.PostConstruct;
|
||||||
|
import javax.annotation.PreDestroy;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author : xiaohai
|
||||||
|
* @date : 2020/08/14 16:20
|
||||||
|
*/
|
||||||
|
@Component
|
||||||
|
public class RedisServerMock {
|
||||||
|
|
||||||
|
private RedisServer redisServer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 构造方法之后执行.
|
||||||
|
*
|
||||||
|
* @throws IOException e
|
||||||
|
*/
|
||||||
|
@PostConstruct
|
||||||
|
public void startRedis() throws IOException {
|
||||||
|
redisServer = new RedisServer(6380);
|
||||||
|
redisServer.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 析构方法之后执行.
|
||||||
|
*/
|
||||||
|
@PreDestroy
|
||||||
|
public void stopRedis() {
|
||||||
|
redisServer.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,18 +4,34 @@ import cn.celess.blog.BaseTest;
|
|||||||
import cn.celess.blog.entity.PartnerSite;
|
import cn.celess.blog.entity.PartnerSite;
|
||||||
import cn.celess.blog.entity.Response;
|
import cn.celess.blog.entity.Response;
|
||||||
import cn.celess.blog.entity.model.PageData;
|
import cn.celess.blog.entity.model.PageData;
|
||||||
|
import cn.celess.blog.entity.request.LinkApplyReq;
|
||||||
import cn.celess.blog.entity.request.LinkReq;
|
import cn.celess.blog.entity.request.LinkReq;
|
||||||
|
import cn.celess.blog.exception.MyException;
|
||||||
import cn.celess.blog.mapper.PartnerMapper;
|
import cn.celess.blog.mapper.PartnerMapper;
|
||||||
import com.fasterxml.jackson.core.type.TypeReference;
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import cn.celess.blog.service.MailService;
|
||||||
|
import cn.celess.blog.service.PartnerSiteService;
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import net.sf.json.JSONObject;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.mail.SimpleMailMessage;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
import static cn.celess.blog.enmu.ResponseEnum.*;
|
import static cn.celess.blog.enmu.ResponseEnum.*;
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
|
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
|
||||||
|
|
||||||
|
@Slf4j
|
||||||
public class LinksControllerTest extends BaseTest {
|
public class LinksControllerTest extends BaseTest {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
@@ -26,6 +42,9 @@ public class LinksControllerTest extends BaseTest {
|
|||||||
};
|
};
|
||||||
private static final TypeReference<?> LINK_MODEL_PAGE_TYPE = new TypeReference<Response<PageData<PartnerSite>>>() {
|
private static final TypeReference<?> LINK_MODEL_PAGE_TYPE = new TypeReference<Response<PageData<PartnerSite>>>() {
|
||||||
};
|
};
|
||||||
|
@Autowired
|
||||||
|
PartnerSiteService partnerSiteService;
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void create() throws Exception {
|
public void create() throws Exception {
|
||||||
@@ -148,15 +167,83 @@ public class LinksControllerTest extends BaseTest {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 手动测试
|
@Test
|
||||||
// @Test
|
public void apply() {
|
||||||
public void apply() throws Exception {
|
// 做service 层的测试
|
||||||
long l = System.currentTimeMillis();
|
// mockEmailServiceInstance(partnerSiteService, "mailService");
|
||||||
String url = "https://www.example.com";
|
mockInjectInstance(partnerSiteService, "mailService", new TestMailServiceImpl());
|
||||||
getMockData(post("/apply?name=小海博客Api测试,请忽略&url=" + url)).andDo(result -> assertEquals(SUCCESS.getCode(), getResponse(result, OBJECT_TYPE).getCode()));
|
LinkApplyReq req = new LinkApplyReq();
|
||||||
System.out.println("耗时:" + (System.currentTimeMillis() - l) / 1000 + "s");
|
req.setName(randomStr(4));
|
||||||
url = "xxx";
|
req.setUrl("https://" + randomStr(4) + ".celess.cn");
|
||||||
getMockData(post("/apply?name=小海博客Api测试,请忽略&url=" + url)).andDo(result -> assertEquals(PARAMETERS_URL_ERROR.getCode(), getResponse(result, OBJECT_TYPE).getCode()));
|
req.setIconPath("https://www.celess.cn/example.png");
|
||||||
|
req.setDesc("desc :" + randomStr());
|
||||||
|
req.setEmail(randomStr(4) + "@celess.cn");
|
||||||
|
req.setLinkUrl(req.getUrl() + "/links");
|
||||||
|
try {
|
||||||
|
// 抓取不到数据的链接
|
||||||
|
partnerSiteService.apply(req);
|
||||||
|
} catch (MyException e) {
|
||||||
|
log.debug("测试抓取不到数据");
|
||||||
|
assertEquals(CANNOT_GET_DATA.getCode(), e.getCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
req.setLinkUrl("https://bing.com");
|
||||||
|
req.setUrl(req.getLinkUrl());
|
||||||
|
try {
|
||||||
|
partnerSiteService.apply(req);
|
||||||
|
} catch (MyException e) {
|
||||||
|
log.debug("测试未添加本站链接的友链申请");
|
||||||
|
assertEquals(APPLY_LINK_NO_ADD_THIS_SITE.getCode(), e.getCode());
|
||||||
|
assertNotNull(e.getResult());
|
||||||
|
try {
|
||||||
|
// 测试uuid一致性
|
||||||
|
log.debug("测试uuid一致性");
|
||||||
|
partnerSiteService.apply(req);
|
||||||
|
} catch (MyException e2) {
|
||||||
|
assertEquals(e.getResult(), e2.getResult());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.debug("测试正常申请");
|
||||||
|
req.setLinkUrl("https://www.celess.cn");
|
||||||
|
req.setUrl(req.getLinkUrl());
|
||||||
|
PartnerSite apply = partnerSiteService.apply(req);
|
||||||
|
assertNotNull(apply);
|
||||||
|
assertNotNull(apply.getId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void reapply() {
|
||||||
|
//mockEmailServiceInstance(partnerSiteService, "mailService");
|
||||||
|
try {
|
||||||
|
partnerSiteService.reapply(randomStr());
|
||||||
|
throw new AssertionError();
|
||||||
|
} catch (MyException e) {
|
||||||
|
assertEquals(DATA_EXPIRED.getCode(), e.getCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
LinkApplyReq req = new LinkApplyReq();
|
||||||
|
req.setName(randomStr(4));
|
||||||
|
req.setIconPath("https://www.celess.cn/example.png");
|
||||||
|
req.setDesc("desc :" + randomStr());
|
||||||
|
req.setEmail(randomStr(4) + "@celess.cn");
|
||||||
|
req.setLinkUrl("https://bing.com");
|
||||||
|
req.setUrl(req.getLinkUrl());
|
||||||
|
String uuid = null;
|
||||||
|
try {
|
||||||
|
partnerSiteService.apply(req);
|
||||||
|
// err here
|
||||||
|
throw new AssertionError();
|
||||||
|
} catch (MyException e) {
|
||||||
|
uuid = (String) e.getResult();
|
||||||
|
String reapply = partnerSiteService.reapply(uuid);
|
||||||
|
assertEquals(reapply, "success");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
partnerSiteService.reapply(uuid);
|
||||||
|
throw new AssertionError();
|
||||||
|
} catch (MyException e) {
|
||||||
|
assertEquals(DATA_EXPIRED.getCode(), e.getCode());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -8,6 +8,7 @@ import cn.celess.blog.entity.model.UserModel;
|
|||||||
import cn.celess.blog.entity.request.LoginReq;
|
import cn.celess.blog.entity.request.LoginReq;
|
||||||
import cn.celess.blog.entity.request.UserReq;
|
import cn.celess.blog.entity.request.UserReq;
|
||||||
import cn.celess.blog.mapper.UserMapper;
|
import cn.celess.blog.mapper.UserMapper;
|
||||||
|
import cn.celess.blog.service.UserService;
|
||||||
import cn.celess.blog.util.MD5Util;
|
import cn.celess.blog.util.MD5Util;
|
||||||
import cn.celess.blog.util.RedisUtil;
|
import cn.celess.blog.util.RedisUtil;
|
||||||
import com.fasterxml.jackson.core.type.TypeReference;
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
@@ -29,12 +30,15 @@ import java.util.stream.Collectors;
|
|||||||
import static cn.celess.blog.enmu.ResponseEnum.*;
|
import static cn.celess.blog.enmu.ResponseEnum.*;
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
|
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
|
||||||
|
import static cn.celess.blog.enmu.ResponseEnum.*;
|
||||||
|
|
||||||
|
|
||||||
public class UserControllerTest extends BaseTest {
|
public class UserControllerTest extends BaseTest {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
UserMapper userMapper;
|
UserMapper userMapper;
|
||||||
@Autowired
|
@Autowired
|
||||||
|
|
||||||
RedisUtil redisUtil;
|
RedisUtil redisUtil;
|
||||||
private static final TypeReference<?> USER_MODEL_TYPE = new TypeReference<Response<UserModel>>() {
|
private static final TypeReference<?> USER_MODEL_TYPE = new TypeReference<Response<UserModel>>() {
|
||||||
};
|
};
|
||||||
@@ -42,6 +46,8 @@ public class UserControllerTest extends BaseTest {
|
|||||||
};
|
};
|
||||||
private static final TypeReference<?> USER_MODEL_LIST_TYPE = new TypeReference<Response<List<Map<String, Object>>>>() {
|
private static final TypeReference<?> USER_MODEL_LIST_TYPE = new TypeReference<Response<List<Map<String, Object>>>>() {
|
||||||
};
|
};
|
||||||
|
UserService userService;
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void login() throws Exception {
|
public void login() throws Exception {
|
||||||
@@ -104,6 +110,10 @@ public class UserControllerTest extends BaseTest {
|
|||||||
connection.setRequestMethod("GET");
|
connection.setRequestMethod("GET");
|
||||||
InputStream inputStream = connection.getInputStream();
|
InputStream inputStream = connection.getInputStream();
|
||||||
assertNotNull(inputStream);
|
assertNotNull(inputStream);
|
||||||
|
|
||||||
|
// mock 实现类
|
||||||
|
mockInjectInstance(userService, "qiniuService", new TestQiNiuServiceImpl());
|
||||||
|
|
||||||
MockMultipartFile file = new MockMultipartFile("file", "logo.png", MediaType.IMAGE_PNG_VALUE, inputStream);
|
MockMultipartFile file = new MockMultipartFile("file", "logo.png", MediaType.IMAGE_PNG_VALUE, inputStream);
|
||||||
getMockData(multipart("/user/imgUpload").file(file), userLogin()).andDo(result -> {
|
getMockData(multipart("/user/imgUpload").file(file), userLogin()).andDo(result -> {
|
||||||
Response<Object> response = getResponse(result, OBJECT_TYPE);
|
Response<Object> response = getResponse(result, OBJECT_TYPE);
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ public class PartnerMapperTest extends BaseTest {
|
|||||||
partnerSite.setIconPath(randomStr(5));
|
partnerSite.setIconPath(randomStr(5));
|
||||||
partnerSite.setDesc(randomStr(5));
|
partnerSite.setDesc(randomStr(5));
|
||||||
partnerSite.setOpen(false);
|
partnerSite.setOpen(false);
|
||||||
partnerSite.setUrl("www.celess.cn?random=" + randomStr(4));
|
partnerSite.setUrl("www.celess.cn/?random=" + randomStr(4));
|
||||||
assertEquals(1, partnerMapper.update(partnerSite));
|
assertEquals(1, partnerMapper.update(partnerSite));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
22
src/test/java/cn/celess/blog/util/HttpUtilTest.java
Normal file
22
src/test/java/cn/celess/blog/util/HttpUtilTest.java
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
package cn.celess.blog.util;
|
||||||
|
|
||||||
|
import cn.celess.blog.BaseTest;
|
||||||
|
import cn.celess.blog.enmu.ResponseEnum;
|
||||||
|
import cn.celess.blog.entity.Response;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
public class HttpUtilTest extends BaseTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void get() {
|
||||||
|
String s = HttpUtil.get("https://api.celess.cn/headerInfo");
|
||||||
|
assertNotNull(s);
|
||||||
|
// Response<Map<String, Object>> response = getResponse(s, MAP_OBJECT_TYPE);
|
||||||
|
// assertEquals(ResponseEnum.SUCCESS.getCode(), response.getCode());
|
||||||
|
// assertNotEquals(0, response.getResult().size());
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user