xxl-job

jobinfo.index.1.js 17KB

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