xxl-job

jobinfo.index.1.js 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. $(function() {
  2. // init date tables
  3. var jobTable = $("#job_list").dataTable({
  4. "deferRender": true,
  5. "processing" : true,
  6. "serverSide": true,
  7. "ajax": {
  8. url: base_url + "/jobinfo/pageList",
  9. data : function ( d ) {
  10. d.jobGroup = $('#jobGroup').val();
  11. d.jobName = $('#jobName').val();
  12. }
  13. },
  14. "searching": false,
  15. "ordering": false,
  16. //"scrollX": true, // X轴滚动条,取消自适应
  17. "columns": [
  18. { "data": 'id', "bSortable": false, "visible" : false},
  19. {
  20. "data": 'jobGroup',
  21. "visible" : false,
  22. "render": function ( data, type, row ) {
  23. var groupMenu = $("#jobGroup").find("option");
  24. for ( var index in $("#jobGroup").find("option")) {
  25. if ($(groupMenu[index]).attr('value') == data) {
  26. return $(groupMenu[index]).html();
  27. }
  28. }
  29. return data;
  30. }
  31. },
  32. { "data": 'jobName'},
  33. { "data": 'jobDesc', "visible" : true},
  34. { "data": 'jobCron', "visible" : true},
  35. { "data": 'jobClass', "visible" : false},
  36. {
  37. "data": 'jobData',
  38. "visible" : true,
  39. "render": function ( data, type, row ) {
  40. var _jobData = eval('(' + data + ')'); // row.jobData
  41. var html = "<p title='" + data + "'>执行器:" + _jobData.handler_name +
  42. "<br>执行参数:" + _jobData.handler_params +
  43. "<br>执行机器:" + _jobData.handler_address + "</p>";
  44. return html;
  45. }
  46. },
  47. {
  48. "data": 'addTime',
  49. "visible" : false,
  50. "render": function ( data, type, row ) {
  51. return data?moment(new Date(data)).format("YYYY-MM-DD HH:mm:ss"):"";
  52. }
  53. },
  54. {
  55. "data": 'updateTime',
  56. "visible" : false,
  57. "render": function ( data, type, row ) {
  58. return data?moment(new Date(data)).format("YYYY-MM-DD HH:mm:ss"):"";
  59. }
  60. },
  61. { "data": 'author', "visible" : true},
  62. { "data": 'alarmEmail', "visible" : false},
  63. { "data": 'alarmThreshold', "visible" : false},
  64. {
  65. "data": 'jobStatus',
  66. "visible" : true,
  67. "render": function ( data, type, row ) {
  68. if ('NORMAL' == data) {
  69. return '<small class="label label-success" ><i class="fa fa-clock-o"></i>'+ data +'</small>';
  70. } else if ('PAUSED' == data){
  71. return '<small class="label label-default" title="暂停" ><i class="fa fa-clock-o"></i>'+ data +'</small>';
  72. } else if ('BLOCKED' == data){
  73. return '<small class="label label-default" title="阻塞[串行]" ><i class="fa fa-clock-o"></i>'+ data +'</small>';
  74. }
  75. return data;
  76. }
  77. },
  78. { "data": '操作' ,
  79. "render": function ( data, type, row ) {
  80. return function(){
  81. // status
  82. var pause_resume = "";
  83. if ('NORMAL' == row.jobStatus) {
  84. pause_resume = '<button class="btn btn-info btn-xs job_operate" type="job_pause" type="button">暂停</button> ';
  85. } else if ('PAUSED' == row.jobStatus){
  86. pause_resume = '<button class="btn btn-info btn-xs job_operate" type="job_resume" type="button">恢复</button> ';
  87. }
  88. // log url
  89. var logUrl = base_url +'/joblog?jobGroup='+ row.jobGroup +'&jobName='+ row.jobName;
  90. // job data
  91. var jobDataMap = eval('(' + row.jobData + ')');
  92. var html = '<p id="'+ row.id +'" '+
  93. ' jobGroup="'+ row.jobGroup +'" '+
  94. ' jobName="'+ row.jobName +'" '+
  95. ' jobCron="'+ row.jobCron +'" '+
  96. ' jobDesc="'+ row.jobDesc +'" '+
  97. ' jobClass="'+ row.jobClass +'" '+
  98. ' jobData="'+ row.jobData +'" '+
  99. ' author="'+ row.author +'" '+
  100. ' alarmEmail="'+ row.alarmEmail +'" '+
  101. ' alarmThreshold="'+ row.alarmThreshold +'" '+
  102. ' handler_params="'+jobDataMap.handler_params +'" '+
  103. ' handler_address="'+ jobDataMap.handler_address +'" '+
  104. ' handler_name="'+ jobDataMap.handler_name +'" '+
  105. '>'+
  106. pause_resume +
  107. '<button class="btn btn-info btn-xs job_operate" type="job_trigger" type="button">执行</button>'+
  108. '<button class="btn btn-warning btn-xs update" type="button">编辑</button><br>'+
  109. '<button class="btn btn-warning btn-xs" type="job_del" type="button" '+
  110. 'onclick="javascript:window.open(\'' + logUrl + '\')" >查看日志</button> '+
  111. '<button class="btn btn-danger btn-xs job_operate" type="job_del" type="button">删除</button> '+
  112. '</p>';
  113. return html;
  114. };
  115. }
  116. }
  117. ],
  118. "language" : {
  119. "sProcessing" : "处理中...",
  120. "sLengthMenu" : "每页 _MENU_ 条记录",
  121. "sZeroRecords" : "没有匹配结果",
  122. "sInfo" : "第 _PAGE_ 页 ( 总共 _PAGES_ 页 )",
  123. "sInfoEmpty" : "无记录",
  124. "sInfoFiltered" : "(由 _MAX_ 项结果过滤)",
  125. "sInfoPostFix" : "",
  126. "sSearch" : "搜索:",
  127. "sUrl" : "",
  128. "sEmptyTable" : "表中数据为空",
  129. "sLoadingRecords" : "载入中...",
  130. "sInfoThousands" : ",",
  131. "oPaginate" : {
  132. "sFirst" : "首页",
  133. "sPrevious" : "上页",
  134. "sNext" : "下页",
  135. "sLast" : "末页"
  136. },
  137. "oAria" : {
  138. "sSortAscending" : ": 以升序排列此列",
  139. "sSortDescending" : ": 以降序排列此列"
  140. }
  141. }
  142. });
  143. // 搜索按钮
  144. $('#searchBtn').on('click', function(){
  145. jobTable.fnDraw();
  146. });
  147. // job operate
  148. $("#job_list").on('click', '.job_operate',function() {
  149. var typeName;
  150. var url;
  151. var type = $(this).attr("type");
  152. if ("job_pause" == type) {
  153. typeName = "暂停";
  154. url = base_url + "/jobinfo/pause";
  155. } else if ("job_resume" == type) {
  156. typeName = "恢复";
  157. url = base_url + "/jobinfo/resume";
  158. } else if ("job_del" == type) {
  159. typeName = "删除";
  160. url = base_url + "/jobinfo/remove";
  161. } else if ("job_trigger" == type) {
  162. typeName = "执行";
  163. url = base_url + "/jobinfo/trigger";
  164. } else {
  165. return;
  166. }
  167. var jobGroup = $(this).parent('p').attr("jobGroup");
  168. var jobName = $(this).parent('p').attr("jobName");
  169. ComConfirm.show("确认" + typeName + "?", function(){
  170. $.ajax({
  171. type : 'POST',
  172. url : url,
  173. data : {
  174. "jobGroup" : jobGroup,
  175. "jobName" : jobName
  176. },
  177. dataType : "json",
  178. success : function(data){
  179. if (data.code == 200) {
  180. ComAlert.show(1, typeName + "成功", function(){
  181. //window.location.reload();
  182. jobTable.fnDraw();
  183. });
  184. } else {
  185. ComAlert.show(1, typeName + "失败");
  186. }
  187. },
  188. });
  189. });
  190. });
  191. // jquery.validate 自定义校验 “英文字母开头,只含有英文字母、数字和下划线”
  192. jQuery.validator.addMethod("myValid01", function(value, element) {
  193. var length = value.length;
  194. var valid = /^[a-zA-Z][a-zA-Z0-9_]*$/;
  195. return this.optional(element) || valid.test(value);
  196. }, "只支持英文字母开头,只含有英文字母、数字和下划线");
  197. // 新增
  198. $(".add").click(function(){
  199. $('#addModal').modal({backdrop: false, keyboard: false}).modal('show');
  200. });
  201. var addModalValidate = $("#addModal .form").validate({
  202. errorElement : 'span',
  203. errorClass : 'help-block',
  204. focusInvalid : true,
  205. rules : {
  206. jobName : {
  207. required : true ,
  208. minlength: 4,
  209. maxlength: 100,
  210. myValid01:true
  211. },
  212. jobCron : {
  213. required : true ,
  214. maxlength: 100
  215. },
  216. jobDesc : {
  217. required : true ,
  218. maxlength: 200
  219. },
  220. handler_address : {
  221. required : true ,
  222. maxlength: 200
  223. },
  224. handler_name : {
  225. required : true ,
  226. maxlength: 200
  227. },
  228. author : {
  229. required : true ,
  230. maxlength: 200
  231. },
  232. alarmEmail : {
  233. required : true ,
  234. maxlength: 200
  235. },
  236. alarmThreshold : {
  237. required : true ,
  238. digits:true
  239. }
  240. },
  241. messages : {
  242. jobName : {
  243. required :"请输入“任务名”" ,
  244. minlength:"“任务名”长度不应低于4位",
  245. maxlength:"“任务名”长度不应超过100位"
  246. },
  247. jobCron : {
  248. required :"请输入“Corn”." ,
  249. maxlength:"“Corn”长度不应超过100位"
  250. },
  251. jobDesc : {
  252. required :"请输入“任务描述”." ,
  253. maxlength:"“任务描述”长度不应超过200位"
  254. },
  255. handler_address : {
  256. required :"请输入“远程-机器地址”." ,
  257. maxlength:"“远程-机器地址”长度不应超过200位"
  258. },
  259. handler_name : {
  260. required : "请输入“远程-执行器”." ,
  261. maxlength: "“远程-执行器”长度不应超过200位"
  262. },
  263. author : {
  264. required : "请输入“负责人”." ,
  265. maxlength: "“负责人”长度不应超过50位"
  266. },
  267. alarmEmail : {
  268. required : "请输入“报警邮件”." ,
  269. maxlength: "“报警邮件”长度不应超过200位"
  270. },
  271. alarmThreshold : {
  272. required : "请输入“报警阈值”." ,
  273. digits:"阀值应该为整数."
  274. }
  275. },
  276. highlight : function(element) {
  277. $(element).closest('.form-group').addClass('has-error');
  278. },
  279. success : function(label) {
  280. label.closest('.form-group').removeClass('has-error');
  281. label.remove();
  282. },
  283. errorPlacement : function(error, element) {
  284. element.parent('div').append(error);
  285. },
  286. submitHandler : function(form) {
  287. $.post(base_url + "/jobinfo/add", $("#addModal .form").serialize(), function(data, status) {
  288. if (data.code == "200") {
  289. ComAlert.show(1, "新增任务成功", function(){
  290. window.location.reload();
  291. });
  292. } else {
  293. if (data.msg) {
  294. ComAlert.show(2, data.msg);
  295. } else {
  296. ComAlert.show(2, "新增失败");
  297. }
  298. }
  299. });
  300. }
  301. });
  302. $("#addModal").on('hide.bs.modal', function () {
  303. $("#addModal .form")[0].reset();
  304. addModalValidate.resetForm();
  305. $("#addModal .form .form-group").removeClass("has-error");
  306. $(".remote_panel").show(); // remote
  307. });
  308. // 远程任务/本地任务,切换
  309. $("#addModal select[name='jobClass']").change(function() {
  310. //console.log( $(this).val().indexOf('RemoteHttpJobBean') );
  311. if($(this).val().indexOf('RemoteHttpJobBean') > -1){
  312. $(".remote_panel").show(); // remote
  313. } else if($(this).val().indexOf('RemoteHttpJobBean') == -1){
  314. $(".remote_panel").hide(); // local
  315. }
  316. });
  317. // 更新
  318. $("#job_list").on('click', '.update',function() {
  319. $("#updateModal .form input[name='jobGroup']").val($(this).parent('p').attr("jobGroup"));
  320. $("#updateModal .form input[name='jobName']").val($(this).parent('p').attr("jobName"));
  321. $("#updateModal .form input[name='jobCron']").val($(this).parent('p').attr("jobCron"));
  322. $("#updateModal .form input[name='jobDesc']").val($(this).parent('p').attr("jobDesc"));
  323. $("#updateModal .form input[name='jobClass']").val($(this).parent('p').attr("jobClass"));
  324. $("#updateModal .form input[name='handler_params']").val($(this).parent('p').attr("handler_params"));
  325. $("#updateModal .form input[name='handler_address']").val($(this).parent('p').attr("handler_address"));
  326. $("#updateModal .form input[name='handler_name']").val($(this).parent('p').attr("handler_name"));
  327. $("#updateModal .form input[name='author']").val($(this).parent('p').attr("author"));
  328. $("#updateModal .form input[name='alarmEmail']").val($(this).parent('p').attr("alarmEmail"));
  329. $("#updateModal .form input[name='alarmThreshold']").val($(this).parent('p').attr("alarmThreshold"));
  330. var _jobClass = $(this).parent('p').attr("jobClass");
  331. if(_jobClass.indexOf('RemoteHttpJobBean') > -1){
  332. $(".remote_panel").show(); // remote
  333. } else if($(this).val().indexOf('RemoteHttpJobBean') == -1){
  334. $(".remote_panel").hide(); // local
  335. }
  336. $('#updateModal').modal({backdrop: false, keyboard: false}).modal('show');
  337. });
  338. var updateModalValidate = $("#updateModal .form").validate({
  339. errorElement : 'span',
  340. errorClass : 'help-block',
  341. focusInvalid : true,
  342. rules : {
  343. jobCron : {
  344. required : true ,
  345. maxlength: 100
  346. },
  347. jobDesc : {
  348. required : true ,
  349. maxlength: 200
  350. },
  351. handler_address : {
  352. required : true ,
  353. maxlength: 200
  354. },
  355. handler_name : {
  356. required : true ,
  357. maxlength: 200
  358. },
  359. author : {
  360. required : true ,
  361. maxlength: 200
  362. },
  363. alarmEmail : {
  364. required : true ,
  365. maxlength: 200
  366. },
  367. alarmThreshold : {
  368. required : true ,
  369. digits:true
  370. }
  371. },
  372. messages : {
  373. jobCron : {
  374. required :"请输入“Corn”." ,
  375. maxlength:"“Corn”长度不应超过100位"
  376. },
  377. jobDesc : {
  378. required :"请输入“任务描述”." ,
  379. maxlength:"“任务描述”长度不应超过200位"
  380. },
  381. handler_address : {
  382. required :"请输入“远程-机器地址”." ,
  383. maxlength:"“远程-机器地址”长度不应超过200位"
  384. },
  385. handler_name : {
  386. required : "请输入“远程-执行器”." ,
  387. maxlength: "“远程-执行器”长度不应超过200位"
  388. },
  389. author : {
  390. required : "请输入“负责人”." ,
  391. maxlength: "“负责人”长度不应超过50位"
  392. },
  393. alarmEmail : {
  394. required : "请输入“报警邮件”." ,
  395. maxlength: "“报警邮件”长度不应超过200位"
  396. },
  397. alarmThreshold : {
  398. required : "请输入“报警阈值”." ,
  399. digits:"阀值应该为整数."
  400. }
  401. },
  402. highlight : function(element) {
  403. $(element).closest('.form-group').addClass('has-error');
  404. },
  405. success : function(label) {
  406. label.closest('.form-group').removeClass('has-error');
  407. label.remove();
  408. },
  409. errorPlacement : function(error, element) {
  410. element.parent('div').append(error);
  411. },
  412. submitHandler : function(form) {
  413. $.post(base_url + "/jobinfo/reschedule", $("#updateModal .form").serialize(), function(data, status) {
  414. if (data.code == "200") {
  415. ComAlert.show(1, "更新成功", function(){
  416. window.location.reload();
  417. });
  418. } else {
  419. if (data.msg) {
  420. ComAlert.show(2, data.msg);
  421. } else {
  422. ComAlert.show(2, "更新失败");
  423. }
  424. }
  425. });
  426. }
  427. });
  428. $("#updateModal").on('hide.bs.modal', function () {
  429. $("#updateModal .form")[0].reset()
  430. });
  431. /*
  432. // 新增-添加参数
  433. $("#addModal .addParam").on('click', function () {
  434. var html = '<div class="form-group newParam">'+
  435. '<label for="lastname" class="col-sm-2 control-label">参数&nbsp;<button class="btn btn-danger btn-xs removeParam" type="button">移除</button></label>'+
  436. '<div class="col-sm-4"><input type="text" class="form-control" name="key" placeholder="请输入参数key[将会强转为String]" maxlength="200" /></div>'+
  437. '<div class="col-sm-6"><input type="text" class="form-control" name="value" placeholder="请输入参数value[将会强转为String]" maxlength="200" /></div>'+
  438. '</div>';
  439. $(this).parents('.form-group').parent().append(html);
  440. $("#addModal .removeParam").on('click', function () {
  441. $(this).parents('.form-group').remove();
  442. });
  443. });
  444. */
  445. });