Selaa lähdekoodia

定时任务目标字符串验证包名白名单

RuoYi 4 vuotta sitten
vanhempi
commit
c44cf9b9f6

+ 11 - 1
ruoyi-common/ruoyi-common-core/src/main/java/com/ruoyi/common/core/constant/Constants.java

@@ -28,6 +28,11 @@ public class Constants
28 28
     public static final String LOOKUP_LDAP = "ldap:";
29 29
 
30 30
     /**
31
+     * LDAPS 远程方法调用
32
+     */
33
+    public static final String LOOKUP_LDAPS = "ldaps:";
34
+
35
+    /**
31 36
      * http请求
32 37
      */
33 38
     public static final String HTTP = "http://";
@@ -114,8 +119,13 @@ public class Constants
114 119
     public static final String RESOURCE_PREFIX = "/profile";
115 120
 
116 121
     /**
122
+     * 定时任务白名单配置(仅允许访问的包名,如其他需要可以自行添加)
123
+     */
124
+    public static final String[] JOB_WHITELIST_STR = { "com.ruoyi" };
125
+
126
+    /**
117 127
      * 定时任务违规的字符
118 128
      */
119 129
     public static final String[] JOB_ERROR_STR = { "java.net.URL", "javax.naming.InitialContext", "org.yaml.snakeyaml",
120
-            "org.springframework" };
130
+            "org.springframework", "org.apache" };
121 131
 }

+ 17 - 8
ruoyi-modules/ruoyi-job/src/main/java/com/ruoyi/job/controller/SysJobController.java

@@ -26,6 +26,7 @@ import com.ruoyi.common.security.utils.SecurityUtils;
26 26
 import com.ruoyi.job.domain.SysJob;
27 27
 import com.ruoyi.job.service.ISysJobService;
28 28
 import com.ruoyi.job.util.CronUtils;
29
+import com.ruoyi.job.util.ScheduleUtils;
29 30
 
30 31
 /**
31 32
  * 调度任务信息操作处理
@@ -88,20 +89,24 @@ public class SysJobController extends BaseController
88 89
         }
89 90
         else if (StringUtils.containsIgnoreCase(job.getInvokeTarget(), Constants.LOOKUP_RMI))
90 91
         {
91
-            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'rmi:'调用");
92
+            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'rmi'调用");
92 93
         }
93
-        else if (StringUtils.containsIgnoreCase(job.getInvokeTarget(), Constants.LOOKUP_LDAP))
94
+        else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[] { Constants.LOOKUP_LDAP, Constants.LOOKUP_LDAPS }))
94 95
         {
95
-            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'ldap:'调用");
96
+            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'ldap'调用");
96 97
         }
97 98
         else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[] { Constants.HTTP, Constants.HTTPS }))
98 99
         {
99
-            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'http(s)//'调用");
100
+            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'http(s)'调用");
100 101
         }
101 102
         else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), Constants.JOB_ERROR_STR))
102 103
         {
103 104
             return error("新增任务'" + job.getJobName() + "'失败,目标字符串存在违规");
104 105
         }
106
+        else if (!ScheduleUtils.whiteList(job.getInvokeTarget()))
107
+        {
108
+            return error("新增任务'" + job.getJobName() + "'失败,目标字符串不在白名单内");
109
+        }
105 110
         job.setCreateBy(SecurityUtils.getUsername());
106 111
         return toAjax(jobService.insertJob(job));
107 112
     }
@@ -120,20 +125,24 @@ public class SysJobController extends BaseController
120 125
         }
121 126
         else if (StringUtils.containsIgnoreCase(job.getInvokeTarget(), Constants.LOOKUP_RMI))
122 127
         {
123
-            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'rmi:'调用");
128
+            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'rmi'调用");
124 129
         }
125
-        else if (StringUtils.containsIgnoreCase(job.getInvokeTarget(), Constants.LOOKUP_LDAP))
130
+        else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[] { Constants.LOOKUP_LDAP, Constants.LOOKUP_LDAPS }))
126 131
         {
127
-            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'ldap:'调用");
132
+            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'ldap'调用");
128 133
         }
129 134
         else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[] { Constants.HTTP, Constants.HTTPS }))
130 135
         {
131
-            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'http(s)//'调用");
136
+            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'http(s)'调用");
132 137
         }
133 138
         else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), Constants.JOB_ERROR_STR))
134 139
         {
135 140
             return error("修改任务'" + job.getJobName() + "'失败,目标字符串存在违规");
136 141
         }
142
+        else if (!ScheduleUtils.whiteList(job.getInvokeTarget()))
143
+        {
144
+            return error("修改任务'" + job.getJobName() + "'失败,目标字符串不在白名单内");
145
+        }
137 146
         job.setUpdateBy(SecurityUtils.getUsername());
138 147
         return toAjax(jobService.updateJob(job));
139 148
     }

+ 23 - 1
ruoyi-modules/ruoyi-job/src/main/java/com/ruoyi/job/util/ScheduleUtils.java

@@ -10,9 +10,11 @@ import org.quartz.Scheduler;
10 10
 import org.quartz.SchedulerException;
11 11
 import org.quartz.TriggerBuilder;
12 12
 import org.quartz.TriggerKey;
13
+import com.ruoyi.common.core.constant.Constants;
13 14
 import com.ruoyi.common.core.constant.ScheduleConstants;
14 15
 import com.ruoyi.common.core.exception.job.TaskException;
15 16
 import com.ruoyi.common.core.exception.job.TaskException.Code;
17
+import com.ruoyi.common.core.utils.StringUtils;
16 18
 import com.ruoyi.job.domain.SysJob;
17 19
 
18 20
 /**
@@ -110,4 +112,24 @@ public class ScheduleUtils
110 112
                         + "' cannot be used in cron schedule tasks", Code.CONFIG_ERROR);
111 113
         }
112 114
     }
113
-}
115
+
116
+    /**
117
+     * 检查包名是否为白名单配置
118
+     * 
119
+     * @param invokeTarget 目标字符串
120
+     * @return 结果
121
+     */
122
+    public static boolean whiteList(String invokeTarget)
123
+    {
124
+        String packageName = StringUtils.substringBefore(invokeTarget, ")");
125
+        int count = StringUtils.countMatches(packageName, ".");
126
+        if (count > 1)
127
+        {
128
+            if (!StringUtils.containsAnyIgnoreCase(invokeTarget, Constants.JOB_WHITELIST_STR))
129
+            {
130
+                return false;
131
+            }
132
+        }
133
+        return true;
134
+    }
135
+}