jobinfo.index.1.js 15 KB

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