-
Notifications
You must be signed in to change notification settings - Fork 261
/
DemoApplication.java
executable file
·291 lines (243 loc) · 11 KB
/
DemoApplication.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
/*Copyright ©2016 TommyLemon(https://github.com/TommyLemon/APIJSON)
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.*/
package apijson.boot;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.serializer.PropertyFilter;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.boot.web.servlet.server.ConfigurableServletWebServerFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import java.lang.reflect.Modifier;
import java.util.List;
import java.util.Map;
import java.util.regex.Pattern;
import javax.naming.Context;
import apijson.Log;
import apijson.NotNull;
import apijson.StringUtil;
import apijson.demo.DemoFunctionParser;
import apijson.demo.DemoParser;
import apijson.demo.DemoSQLConfig;
import apijson.demo.DemoSQLExecutor;
import apijson.demo.DemoVerifier;
import apijson.framework.APIJSONApplication;
import apijson.framework.APIJSONCreator;
import apijson.framework.APIJSONParser;
import apijson.orm.AbstractVerifier;
import apijson.orm.FunctionParser;
import apijson.orm.Parser;
import apijson.orm.SQLConfig;
import apijson.orm.SQLExecutor;
import apijson.orm.Verifier;
import unitauto.MethodUtil;
import unitauto.MethodUtil.Argument;
import unitauto.MethodUtil.InstanceGetter;
import unitauto.MethodUtil.JSONCallback;
import unitauto.jar.UnitAutoApp;
/**
* Demo SpringBoot Application 主应用程序启动类
* 右键这个类 > Run As > Java Application
* 具体见 SpringBoot 文档
* https://www.springcloud.cc/spring-boot.html#using-boot-locating-the-main-class
*
* @author Lemon
*/
@Configuration
@SpringBootApplication
public class DemoApplication implements WebServerFactoryCustomizer<ConfigurableServletWebServerFactory> {
private static final String TAG = "DemoApplication";
// 全局 ApplicationContext 实例,方便 getBean 拿到 Spring/SpringBoot 注入的类实例
private static ApplicationContext APPLICATION_CONTEXT;
public static ApplicationContext getApplicationContext() {
return APPLICATION_CONTEXT;
}
public static void main(String[] args) throws Exception {
APPLICATION_CONTEXT = SpringApplication.run(DemoApplication.class, args);
// FIXME 不要开放给项目组后端之外的任何人使用 UnitAuto(强制登录鉴权)!!!如果不需要单元测试则移除相关代码或 unitauto.Log.DEBUG = false;
// 上线生产环境前改为 false,可不输出 APIJSONORM 的日志 以及 SQLException 的原始(敏感)信息
unitauto.Log.DEBUG = Log.DEBUG = true;
APIJSONParser.IS_PRINT_BIG_LOG = true;
APIJSONApplication.init();
}
// SpringBoot 2.x 自定义端口方式
@Override
public void customize(ConfigurableServletWebServerFactory server) {
server.setPort(8080);
}
// 支持 APIAuto 中 JavaScript 代码跨域请求
@Bean
public WebMvcConfigurer corsConfigurer() {
return new WebMvcConfigurer() {
@Override
public void addCorsMappings(CorsRegistry registry) {
registry.addMapping("/**")
.allowedOriginPatterns("*")
.allowedMethods("*")
.allowCredentials(true)
.exposedHeaders(DemoController.APIJSON_DELEGATE_ID) // Cookie 和 Set-Cookie 怎么设置都没用 ,Cookie,Set-Cookie") // .exposedHeaders("*")
.maxAge(3600);
}
};
}
static {
// 把以下需要用到的数据库驱动取消注释即可,如果这里没有可以自己新增
// try { //加载驱动程序
// Log.d(TAG, "尝试加载 SQLServer 驱动 <<<<<<<<<<<<<<<<<<<<< ");
// Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver");
// Log.d(TAG, "成功加载 SQLServer 驱动!>>>>>>>>>>>>>>>>>>>>> ");
// }
// catch (ClassNotFoundException e) {
// e.printStackTrace();
// Log.e(TAG, "加载 SQLServer 驱动失败,请检查 pom.xml 中 net.sourceforge.jtds 版本是否存在以及可用 !!!");
// }
//
// try { //加载驱动程序
// Log.d(TAG, "尝试加载 Oracle 驱动 <<<<<<<<<<<<<<<<<<<<< ");
// Class.forName("oracle.jdbc.driver.OracleDriver");
// Log.d(TAG, "成功加载 Oracle 驱动!>>>>>>>>>>>>>>>>>>>>> ");
// }
// catch (ClassNotFoundException e) {
// e.printStackTrace();
// Log.e(TAG, "加载 Oracle 驱动失败,请检查 pom.xml 中 com.oracle.jdbc 版本是否存在以及可用 !!!");
// }
//
// try { //加载驱动程序
// Log.d(TAG, "尝试加载 DB2 驱动 <<<<<<<<<<<<<<<<<<<<< ");
// Class.forName("com.ibm.db2.jcc.DB2Driver");
// Log.d(TAG, "成功加载 DB2 驱动!>>>>>>>>>>>>>>>>>>>>> ");
// }
// catch (ClassNotFoundException e) {
// e.printStackTrace();
// Log.e(TAG, "加载 DB2 驱动失败,请检查 pom.xml 中 com.ibm.db2 版本是否存在以及可用 !!!");
// }
try { //加载驱动程序
Log.d(TAG, "尝试加载 TDengine 驱动 <<<<<<<<<<<<<<<<<<<<< ");
Class.forName("com.taosdata.jdbc.TSDBDriver");
Log.d(TAG, "成功加载 TDengine 驱动!>>>>>>>>>>>>>>>>>>>>> ");
} catch (ClassNotFoundException e) {
e.printStackTrace();
Log.e(TAG, "加载 TDengine 驱动失败,请检查 pom.xml 中 com.taosdata.jdbc 版本是否存在以及可用 !!!");
}
// APIJSON 配置 <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
Map<String, Pattern> COMPILE_MAP = AbstractVerifier.COMPILE_MAP;
COMPILE_MAP.put("PHONE", StringUtil.PATTERN_PHONE);
COMPILE_MAP.put("EMAIL", StringUtil.PATTERN_EMAIL);
COMPILE_MAP.put("ID_CARD", StringUtil.PATTERN_ID_CARD);
// 使用本项目的自定义处理类
APIJSONApplication.DEFAULT_APIJSON_CREATOR = new APIJSONCreator<Long>() {
@Override
public Parser<Long> createParser() {
return new DemoParser();
}
@Override
public FunctionParser createFunctionParser() {
return new DemoFunctionParser();
}
@Override
public Verifier<Long> createVerifier() {
return new DemoVerifier();
}
@Override
public SQLConfig createSQLConfig() {
return new DemoSQLConfig();
}
@Override
public SQLExecutor createSQLExecutor() {
return new DemoSQLExecutor();
}
};
// APIJSON 配置 >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
// UnitAuto 单元测试配置 https://github.com/TommyLemon/UnitAuto <<<<<<<<<<<<<<<<<<<<<<<<<<<
// FIXME 不要开放给项目组后端之外的任何人使用 UnitAuto(强制登录鉴权)!!!如果不需要单元测试则移除相关代码或 unitauto.Log.DEBUG = false;
UnitAutoApp.init();
// 适配 Spring 注入的类及 Context 等环境相关的类
final InstanceGetter ig = MethodUtil.INSTANCE_GETTER;
MethodUtil.INSTANCE_GETTER = new InstanceGetter() {
@Override
public Object getInstance(@NotNull Class<?> clazz, List<Argument> classArgs, Boolean reuse) throws Exception {
if (APPLICATION_CONTEXT != null && ApplicationContext.class.isAssignableFrom(clazz) && clazz.isAssignableFrom(APPLICATION_CONTEXT.getClass())) {
return APPLICATION_CONTEXT;
}
if (reuse != null && reuse && (classArgs == null || classArgs.isEmpty())) {
return APPLICATION_CONTEXT.getBean(clazz);
}
return ig.getInstance(clazz, classArgs, reuse);
}
};
// 排除转换 JSON 异常的类,一般是 Context 等环境相关的类
final JSONCallback jc = MethodUtil.JSON_CALLBACK;
MethodUtil.JSON_CALLBACK = new JSONCallback() {
@Override
public JSONObject newSuccessResult() {
return jc.newSuccessResult();
}
@Override
public JSONObject newErrorResult(Throwable e) {
return jc.newErrorResult(e);
}
@Override
public JSONObject parseJSON(String type, Object value) {
if (value == null || unitauto.JSON.isBooleanOrNumberOrString(value) || value instanceof JSON || value instanceof Enum) {
return jc.parseJSON(type, value);
}
if (value instanceof ApplicationContext
|| value instanceof Context
|| value instanceof org.apache.catalina.Context
// SpringBoot 2.6.7 已移除 || value instanceof ch.qos.logback.core.Context
) {
value = value.toString();
} else {
try {
value = JSON.parse(JSON.toJSONString(value, new PropertyFilter() {
@Override
public boolean apply(Object object, String name, Object value) {
if (value == null) {
return true;
}
if (value instanceof ApplicationContext
|| value instanceof Context
|| value instanceof org.apache.catalina.Context
// SpringBoot 2.6.7 已移除 || value instanceof ch.qos.logback.core.Context
) {
return false;
}
// 防止通过 UnitAuto 远程执行 getDBPassword 等方法来查到敏感信息,但如果直接调用 public String getDBUri 这里没法拦截,仍然会返回敏感信息
// if (object instanceof SQLConfig) {
// // 这个类部分方法不序列化返回
// if ("dBUri".equalsIgnoreCase(name) || "dBPassword".equalsIgnoreCase(name) || "dBAccount".equalsIgnoreCase(name)) {
// return false;
// }
// return false; // 这个类所有方法都不序列化返回
// }
// 所有类中的方法只要包含关键词就不序列化返回
String n = StringUtil.toLowerCase(name);
if (n.contains("database") || n.contains("schema") || n.contains("dburi") || n.contains("password") || n.contains("account")) {
return false;
}
return Modifier.isPublic(value.getClass().getModifiers());
}
}));
} catch (Exception e) {
Log.e(TAG, "toJSONString catch \n" + e.getMessage());
}
}
return jc.parseJSON(type, value);
}
};
// UnitAuto 单元测试配置 https://github.com/TommyLemon/UnitAuto >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
}
}