aind-data-transfer-service 1.12.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of aind-data-transfer-service might be problematic. Click here for more details.
- aind_data_transfer_service/__init__.py +9 -0
- aind_data_transfer_service/configs/__init__.py +1 -0
- aind_data_transfer_service/configs/csv_handler.py +59 -0
- aind_data_transfer_service/configs/job_configs.py +545 -0
- aind_data_transfer_service/configs/job_upload_template.py +153 -0
- aind_data_transfer_service/hpc/__init__.py +1 -0
- aind_data_transfer_service/hpc/client.py +151 -0
- aind_data_transfer_service/hpc/models.py +492 -0
- aind_data_transfer_service/log_handler.py +58 -0
- aind_data_transfer_service/models/__init__.py +1 -0
- aind_data_transfer_service/models/core.py +300 -0
- aind_data_transfer_service/models/internal.py +277 -0
- aind_data_transfer_service/server.py +1125 -0
- aind_data_transfer_service/templates/index.html +245 -0
- aind_data_transfer_service/templates/job_params.html +194 -0
- aind_data_transfer_service/templates/job_status.html +323 -0
- aind_data_transfer_service/templates/job_tasks_table.html +146 -0
- aind_data_transfer_service/templates/task_logs.html +31 -0
- aind_data_transfer_service-1.12.0.dist-info/METADATA +49 -0
- aind_data_transfer_service-1.12.0.dist-info/RECORD +23 -0
- aind_data_transfer_service-1.12.0.dist-info/WHEEL +5 -0
- aind_data_transfer_service-1.12.0.dist-info/licenses/LICENSE +21 -0
- aind_data_transfer_service-1.12.0.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,245 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html>
|
|
3
|
+
<head>
|
|
4
|
+
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.6.4/jquery.min.js"></script>
|
|
5
|
+
<meta charset="UTF-8">
|
|
6
|
+
<title>{% block title %} {% endblock %} AIND Data Transfer Service</title>
|
|
7
|
+
<style>
|
|
8
|
+
body {
|
|
9
|
+
margin: 20px;
|
|
10
|
+
font-family: arial, sans-serif;
|
|
11
|
+
}
|
|
12
|
+
fieldset {
|
|
13
|
+
display:inline
|
|
14
|
+
}
|
|
15
|
+
table {
|
|
16
|
+
border-collapse: collapse;
|
|
17
|
+
width: 75%;
|
|
18
|
+
}
|
|
19
|
+
td, th {
|
|
20
|
+
border: 1px solid #dddddd;
|
|
21
|
+
text-align: left;
|
|
22
|
+
padding: 8px;
|
|
23
|
+
}
|
|
24
|
+
th, tr:hover {
|
|
25
|
+
background-color: #E8EAF6;
|
|
26
|
+
color: #3F51B5;
|
|
27
|
+
}
|
|
28
|
+
#submit {
|
|
29
|
+
border: none;
|
|
30
|
+
border-radius: 6px;
|
|
31
|
+
color: #ffffff;
|
|
32
|
+
padding: 8px;
|
|
33
|
+
width: 80px;
|
|
34
|
+
font-size: medium;
|
|
35
|
+
background-color: #5065df;
|
|
36
|
+
&:hover { background-color: #3f51b5; }
|
|
37
|
+
}
|
|
38
|
+
#message {
|
|
39
|
+
margin-top: 20px;
|
|
40
|
+
&.success { color: green; }
|
|
41
|
+
&.error { color: red; }
|
|
42
|
+
}
|
|
43
|
+
</style>
|
|
44
|
+
</head>
|
|
45
|
+
<body>
|
|
46
|
+
<nav>
|
|
47
|
+
<a href="/">Submit Jobs</a> |
|
|
48
|
+
<a href="/jobs">Job Status</a> |
|
|
49
|
+
<a href="/job_params">Job Parameters</a> |
|
|
50
|
+
<a title="Download job template as .xslx" href= "/api/job_upload_template" download>Job Submit Template</a> |
|
|
51
|
+
<a title="List of project names" href= "{{ project_names_url }}" target="_blank" >Project Names</a> |
|
|
52
|
+
<a title="For more information click here" href="https://aind-data-transfer-service.readthedocs.io" target="_blank" >Help</a>
|
|
53
|
+
</nav>
|
|
54
|
+
<br>
|
|
55
|
+
<div>
|
|
56
|
+
<p1>For instructions on how to submit jobs click on the Help tab</p1>
|
|
57
|
+
</div>
|
|
58
|
+
<h2>Submit Jobs</h2>
|
|
59
|
+
<div>
|
|
60
|
+
<fieldset>
|
|
61
|
+
<legend>Mail Notifications (optional)</legend><br>
|
|
62
|
+
<div>
|
|
63
|
+
<label for="email" title="Optionally provide an allen institute email address to receive upload job status notifications">Allen Institute email:</label>
|
|
64
|
+
<input type="email" id="email" pattern=".+@alleninstitute\.org" size="30" placeholder="@alleninstitute.org" /><br><br>
|
|
65
|
+
</div>
|
|
66
|
+
<div>
|
|
67
|
+
<input type="checkbox" id="begin" name="begin" />
|
|
68
|
+
<label for="begin">BEGIN</label> |
|
|
69
|
+
<input type="checkbox" id="end" name="end" />
|
|
70
|
+
<label for="end">END</label> |
|
|
71
|
+
<input type="checkbox" id="fail" name="fail" checked />
|
|
72
|
+
<label for="fail">FAIL</label> |
|
|
73
|
+
<input type="checkbox" id="retry" name="retry" />
|
|
74
|
+
<label for="retry">RETRY</label> |
|
|
75
|
+
<input type="checkbox" id="all" name="all" />
|
|
76
|
+
<label for="all">ALL</label>
|
|
77
|
+
</div>
|
|
78
|
+
</fieldset>
|
|
79
|
+
</div><br><br>
|
|
80
|
+
<form id="file_form" method="post" enctype="multipart/form-data">
|
|
81
|
+
<label for="file">Please select a .csv or .xlsx file:</label>
|
|
82
|
+
<input type="file" id="file" name="file" accept=".csv,.xlsx" onchange="validateJobs(this)" onclick="this.value=null"><br><br>
|
|
83
|
+
</form>
|
|
84
|
+
<button id="submit" type="button" onclick="submitJobs()">Submit</button>
|
|
85
|
+
<div id="message"></div><br>
|
|
86
|
+
<div id="response"></div>
|
|
87
|
+
<script>
|
|
88
|
+
var jobs = []
|
|
89
|
+
var parsing_errors = []
|
|
90
|
+
const msgTypes = {
|
|
91
|
+
"validatePending": "Sending...",
|
|
92
|
+
"validateSuccess": "Successfully validated jobs from file.",
|
|
93
|
+
"validateError": "Error validating jobs from file.",
|
|
94
|
+
"submitPending": "Submitting jobs. Please do not refresh or re-submit...",
|
|
95
|
+
"submitSuccess": "Successfully submitted jobs.",
|
|
96
|
+
"submitError": "Error submitting jobs."
|
|
97
|
+
}
|
|
98
|
+
setMessage = function(msgType) {
|
|
99
|
+
$("#message").removeClass("success");
|
|
100
|
+
$("#message").removeClass("error");
|
|
101
|
+
$("#message").html(msgType);
|
|
102
|
+
if (msgType == msgTypes.validateSuccess || msgType == msgTypes.submitSuccess) {
|
|
103
|
+
$("#message").addClass("success");
|
|
104
|
+
} else if (msgType == msgTypes.validateError || msgType == msgTypes.submitError) {
|
|
105
|
+
$("#message").addClass("error");
|
|
106
|
+
}
|
|
107
|
+
};
|
|
108
|
+
addTableRow = function(data, table, tr, td, isHeader) {
|
|
109
|
+
tr = document.createElement('tr');
|
|
110
|
+
for (var d of data) {
|
|
111
|
+
td = document.createElement(isHeader ? 'th' : 'td');
|
|
112
|
+
td.innerHTML = d.value ?? d;
|
|
113
|
+
if (d.rowspan) {
|
|
114
|
+
td.setAttribute("rowspan", d.rowspan);
|
|
115
|
+
}
|
|
116
|
+
tr.append(td);
|
|
117
|
+
}
|
|
118
|
+
table.appendChild(tr);
|
|
119
|
+
};
|
|
120
|
+
validateJobs = function(fileElement) {
|
|
121
|
+
if (fileElement.files.length != 1) {
|
|
122
|
+
// File attach was cancelled by user
|
|
123
|
+
return;
|
|
124
|
+
}
|
|
125
|
+
if (![".csv", ".xlsx"].some(ext => fileElement.files[0].name.endsWith(ext))) {
|
|
126
|
+
fileElement.value = null;
|
|
127
|
+
alert("Invalid file type. Please attach a .csv or .xlsx file.");
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
var formData = new FormData(document.getElementById("file_form"));
|
|
131
|
+
$.ajax({
|
|
132
|
+
url: "/api/v1/validate_csv",
|
|
133
|
+
type: "POST",
|
|
134
|
+
data: formData,
|
|
135
|
+
cache: false,
|
|
136
|
+
contentType: false,
|
|
137
|
+
processData: false,
|
|
138
|
+
beforeSend: function() {
|
|
139
|
+
setMessage(msgTypes.validatePending);
|
|
140
|
+
$("#response").html("");
|
|
141
|
+
},
|
|
142
|
+
success: function(data) {
|
|
143
|
+
setMessage(msgTypes.validateSuccess);
|
|
144
|
+
jobs = data["data"]["jobs"];
|
|
145
|
+
parsing_errors = []
|
|
146
|
+
let jobsLength = jobs.length;
|
|
147
|
+
var table = document.createElement('table'), tr, td, row;
|
|
148
|
+
addTableRow(
|
|
149
|
+
[ "project_name", "process_capsule_id", "input_data_mount", "s3_bucket", "platform", "subject_id", "acq_datetime", "metadata_dir", "modality", "modality.source" ],
|
|
150
|
+
table, tr, td, true
|
|
151
|
+
);
|
|
152
|
+
for (row = 0; row < jobsLength; row++) {
|
|
153
|
+
let job = jobs[row];
|
|
154
|
+
let modalities = job.modalities;
|
|
155
|
+
let modalitiesLength = modalities.length;
|
|
156
|
+
addTableRow(
|
|
157
|
+
[ { value: job.project_name, rowspan: modalitiesLength },
|
|
158
|
+
{ value: job.process_capsule_id ?? "", rowspan: modalitiesLength },
|
|
159
|
+
{ value: job.input_data_mount ?? "", rowspan: modalitiesLength },
|
|
160
|
+
{ value: job.s3_bucket, rowspan: modalitiesLength },
|
|
161
|
+
{ value: job.platform.abbreviation, rowspan: modalitiesLength },
|
|
162
|
+
{ value: job.subject_id, rowspan: modalitiesLength },
|
|
163
|
+
{ value: job.acq_datetime, rowspan: modalitiesLength },
|
|
164
|
+
{ value: job.metadata_dir ?? "", rowspan: modalitiesLength },
|
|
165
|
+
modalities ? modalities[0].modality.abbreviation : "",
|
|
166
|
+
modalities ? modalities[0].source : ""
|
|
167
|
+
], table, tr, td, false
|
|
168
|
+
);
|
|
169
|
+
for (mRow = 1; mRow < modalitiesLength; mRow++) {
|
|
170
|
+
let modality = modalities[mRow]
|
|
171
|
+
addTableRow(
|
|
172
|
+
[ modality.modality.abbreviation, modality.source ],
|
|
173
|
+
table, tr, td, false
|
|
174
|
+
);
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
$("#response").html(table);
|
|
178
|
+
},
|
|
179
|
+
error: function(data) {
|
|
180
|
+
jobs = []
|
|
181
|
+
parsing_errors = data.responseJSON["data"]["errors"]
|
|
182
|
+
setMessage(msgTypes.validateError);
|
|
183
|
+
$("#response").html(parsing_errors.map((err) => {
|
|
184
|
+
return `<li>${err}</li>`
|
|
185
|
+
}));
|
|
186
|
+
}
|
|
187
|
+
});
|
|
188
|
+
};
|
|
189
|
+
submitJobs = function() {
|
|
190
|
+
if(jobs.length > 0 && parsing_errors.length == 0){
|
|
191
|
+
let job_settings = {};
|
|
192
|
+
let mail_user = $("#email").val();
|
|
193
|
+
if (mail_user !== "" && mail_user !== undefined) {
|
|
194
|
+
job_settings["user_email"] = mail_user;
|
|
195
|
+
};
|
|
196
|
+
let mail_type = [];
|
|
197
|
+
if ($("#all").is(":checked")) {
|
|
198
|
+
mail_type = ["all"];
|
|
199
|
+
};
|
|
200
|
+
if ($("#begin").is(":checked")) {
|
|
201
|
+
mail_type.push("begin");
|
|
202
|
+
};
|
|
203
|
+
if ($("#end").is(":checked")) {
|
|
204
|
+
mail_type.push("end");
|
|
205
|
+
};
|
|
206
|
+
if ($("#fail").is(":checked")) {
|
|
207
|
+
mail_type.push("fail");
|
|
208
|
+
};
|
|
209
|
+
if ($("#retry").is(":checked")) {
|
|
210
|
+
mail_type.push("retry");
|
|
211
|
+
};
|
|
212
|
+
job_settings["email_notification_types"] = mail_type
|
|
213
|
+
job_settings["upload_jobs"] = jobs
|
|
214
|
+
$.ajax({
|
|
215
|
+
url: "/api/v1/submit_jobs",
|
|
216
|
+
type: "POST",
|
|
217
|
+
data: JSON.stringify(job_settings),
|
|
218
|
+
contentType: 'application/json; charset=utf-8',
|
|
219
|
+
beforeSend: function() {
|
|
220
|
+
setMessage(msgTypes.submitPending);
|
|
221
|
+
$("#response").html("");
|
|
222
|
+
},
|
|
223
|
+
success: function (data) {
|
|
224
|
+
jobs = []
|
|
225
|
+
parsing_errors = []
|
|
226
|
+
setMessage(msgTypes.submitSuccess);
|
|
227
|
+
$("#response").html(data);
|
|
228
|
+
},
|
|
229
|
+
error: function(data) {
|
|
230
|
+
jobs = []
|
|
231
|
+
setMessage(msgTypes.submitError);
|
|
232
|
+
parsing_errors = data.responseJSON["data"]["errors"]
|
|
233
|
+
$("#response").html(parsing_errors.map((errStr) => {
|
|
234
|
+
return `<li>${errStr}</li>`
|
|
235
|
+
}));
|
|
236
|
+
}
|
|
237
|
+
});
|
|
238
|
+
} else if (jobs.length == 0) {
|
|
239
|
+
alert("No valid jobs to submit. Please attach a .csv or .xlsx file with valid jobs.");
|
|
240
|
+
return;
|
|
241
|
+
}
|
|
242
|
+
};
|
|
243
|
+
</script>
|
|
244
|
+
</body>
|
|
245
|
+
</html>
|
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html>
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8">
|
|
5
|
+
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" rel="stylesheet">
|
|
6
|
+
<link href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap-icons/1.8.1/font/bootstrap-icons.min.css" rel="stylesheet">
|
|
7
|
+
<link rel="stylesheet" href="https://cdn.datatables.net/2.1.8/css/dataTables.dataTables.css" />
|
|
8
|
+
<link rel="stylesheet" href="https://cdn.datatables.net/searchpanes/2.3.3/css/searchPanes.dataTables.css" />
|
|
9
|
+
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/js/bootstrap.bundle.min.js"></script>
|
|
10
|
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.3/moment.min.js"></script>
|
|
11
|
+
<script type="text/javascript" src="https://cdn.jsdelivr.net/jquery/latest/jquery.min.js"></script>
|
|
12
|
+
<script src="https://cdn.datatables.net/2.1.8/js/dataTables.js"></script>
|
|
13
|
+
<script src="https://cdn.datatables.net/searchpanes/2.3.3/js/dataTables.searchPanes.js"></script>
|
|
14
|
+
<script src="https://cdn.datatables.net/select/3.0.0/js/dataTables.select.js"></script>
|
|
15
|
+
<title>{% block title %} {% endblock %} AIND Data Transfer Service Job Parameters</title>
|
|
16
|
+
<style>
|
|
17
|
+
body {
|
|
18
|
+
margin: 20px;
|
|
19
|
+
font-family: arial, sans-serif;
|
|
20
|
+
}
|
|
21
|
+
nav {
|
|
22
|
+
height: 40px;
|
|
23
|
+
}
|
|
24
|
+
.modal-body {
|
|
25
|
+
max-height: calc(100vh - 200px);
|
|
26
|
+
overflow: auto;
|
|
27
|
+
}
|
|
28
|
+
</style>
|
|
29
|
+
</head>
|
|
30
|
+
<body>
|
|
31
|
+
<nav>
|
|
32
|
+
<a href="/">Submit Jobs</a> |
|
|
33
|
+
<a href="/jobs">Job Status</a> |
|
|
34
|
+
<a href="/job_params">Job Parameters</a> |
|
|
35
|
+
<a title="Download job template as .xslx" href= "/api/job_upload_template" download>Job Submit Template</a> |
|
|
36
|
+
<a title="List of project names" href= "{{ project_names_url }}" target="_blank" >Project Names</a> |
|
|
37
|
+
<a title="For more information click here" href="https://aind-data-transfer-service.readthedocs.io" target="_blank" >Help</a>
|
|
38
|
+
</nav>
|
|
39
|
+
<div class="content">
|
|
40
|
+
<h4 class="mb-2">
|
|
41
|
+
<!-- dropdown to switch version -->
|
|
42
|
+
<div id="version-dropdown" class="btn-group mb-2">
|
|
43
|
+
<button id="version-button" type="button" class="btn btn-primary dropdown-toggle" data-bs-toggle="dropdown">
|
|
44
|
+
{{default_version}}
|
|
45
|
+
</button>
|
|
46
|
+
<ul class="dropdown-menu">
|
|
47
|
+
{% for v in versions %}
|
|
48
|
+
<button class="dropdown-item" type="button">{{ v }}</button>
|
|
49
|
+
{% endfor %}
|
|
50
|
+
</ul>
|
|
51
|
+
</div>
|
|
52
|
+
<span>Job Parameters</span>
|
|
53
|
+
</h4>
|
|
54
|
+
<!-- job params table -->
|
|
55
|
+
<div>
|
|
56
|
+
<table id="job-params-table" class="display compact table table-bordered table-sm" style="font-size: small">
|
|
57
|
+
<thead>
|
|
58
|
+
<tr>
|
|
59
|
+
<th>Job Type</th>
|
|
60
|
+
<th>Task ID</th>
|
|
61
|
+
<th>Modality</th>
|
|
62
|
+
<th>Parameter Name</th>
|
|
63
|
+
<th>Last Modified</th>
|
|
64
|
+
</tr>
|
|
65
|
+
</thead>
|
|
66
|
+
</table>
|
|
67
|
+
<!-- modal for displaying param value as json -->
|
|
68
|
+
<div class="modal fade" id="param-modal" tabindex="-1" aria-labelledby="param-modal-label" aria-hidden="true">
|
|
69
|
+
<div class="modal-dialog modal-xl">
|
|
70
|
+
<div class="modal-content">
|
|
71
|
+
<div class="modal-header">
|
|
72
|
+
<h5 class="modal-title" id="param-modal-label"></h5>
|
|
73
|
+
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
|
|
74
|
+
</div>
|
|
75
|
+
<div class="modal-body">
|
|
76
|
+
<pre id="param-modal-content" class="bg-light p-1 border rounded"></pre>
|
|
77
|
+
</div>
|
|
78
|
+
</div>
|
|
79
|
+
</div>
|
|
80
|
+
</div>
|
|
81
|
+
</div>
|
|
82
|
+
</div>
|
|
83
|
+
<script>
|
|
84
|
+
$(document).ready(function() {
|
|
85
|
+
createJobParamsTable();
|
|
86
|
+
// Event listeners for param value modal
|
|
87
|
+
$('#param-modal').on('show.bs.modal', function(event) {
|
|
88
|
+
var button = $(event.relatedTarget);
|
|
89
|
+
var paramName = button.data('bs-param-name');
|
|
90
|
+
var jobType = button.data('bs-job-type');
|
|
91
|
+
var taskId = button.data('bs-task-id');
|
|
92
|
+
var modality = button.data('bs-modality');
|
|
93
|
+
// update the modal label and contents
|
|
94
|
+
var modal = $(this);
|
|
95
|
+
modal.find('#param-modal-label').text(paramName);
|
|
96
|
+
var version = $('#version-button').text().trim();
|
|
97
|
+
var getParameterUrl = `/api/${version}/parameters/job_types/${jobType}/tasks/${taskId}`;
|
|
98
|
+
if (modality) {
|
|
99
|
+
getParameterUrl += `/${modality}`;
|
|
100
|
+
}
|
|
101
|
+
$.ajax({
|
|
102
|
+
url: getParameterUrl,
|
|
103
|
+
type: 'GET',
|
|
104
|
+
success: function(response) {
|
|
105
|
+
jsonStr = JSON.stringify(response.data, null, 3);
|
|
106
|
+
modal.find('#param-modal-content').text(jsonStr);
|
|
107
|
+
},
|
|
108
|
+
error: function(xhr, status, error) {
|
|
109
|
+
modal.find('#param-modal-content').text(error);
|
|
110
|
+
}
|
|
111
|
+
});
|
|
112
|
+
});
|
|
113
|
+
$('#param-modal').on('hidden.bs.modal', function() {
|
|
114
|
+
$(this).find('#param-modal-label').text('');
|
|
115
|
+
$(this).find('#param-modal-content').text('');
|
|
116
|
+
});
|
|
117
|
+
// Event listener for version dropdown
|
|
118
|
+
$('#version-dropdown .dropdown-item').on('click', function() {
|
|
119
|
+
var version = $(this).text();
|
|
120
|
+
if (version != $('#version-button').text().trim()) {
|
|
121
|
+
$('#version-button').text(version);
|
|
122
|
+
$('#job-params-table').DataTable().ajax.url(`/api/${version}/parameters`).load();
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
});
|
|
126
|
+
// Create DataTable for job params table
|
|
127
|
+
function createJobParamsTable() {
|
|
128
|
+
$('#job-params-table').DataTable({
|
|
129
|
+
ajax: {
|
|
130
|
+
url: "/api/{{default_version}}/parameters",
|
|
131
|
+
dataSrc: 'data'
|
|
132
|
+
},
|
|
133
|
+
processing: true,
|
|
134
|
+
columns: [
|
|
135
|
+
{ data: 'job_type', searchPanes: {show: true} },
|
|
136
|
+
{ data: 'task_id', searchPanes: {show: true} },
|
|
137
|
+
{ data: 'modality', searchPanes: {show: false} },
|
|
138
|
+
{ data: 'name', render: renderParameterButton },
|
|
139
|
+
{ data: 'last_modified', render: renderDatetime },
|
|
140
|
+
],
|
|
141
|
+
order: [0, 'asc'],
|
|
142
|
+
// preselect "default" job_type
|
|
143
|
+
searchPanes: {
|
|
144
|
+
preSelect: [ { rows: ['default'], column: 0 } ],
|
|
145
|
+
},
|
|
146
|
+
// options to match default jobs table
|
|
147
|
+
pageLength: 25,
|
|
148
|
+
layout: {
|
|
149
|
+
topStart: null,
|
|
150
|
+
topEnd: null,
|
|
151
|
+
bottomStart: null,
|
|
152
|
+
bottomEnd: null,
|
|
153
|
+
top1Start: "searchPanes",
|
|
154
|
+
top: [
|
|
155
|
+
'pageLength',
|
|
156
|
+
'info',
|
|
157
|
+
{ paging: { numbers: false } }
|
|
158
|
+
],
|
|
159
|
+
},
|
|
160
|
+
language: {
|
|
161
|
+
info: "_START_ to _END_ of _TOTAL_",
|
|
162
|
+
infoEmpty: "0 to 0 of 0",
|
|
163
|
+
entries: { _: "parameters", 1: "parameter" },
|
|
164
|
+
paginate: {
|
|
165
|
+
first: '« First',
|
|
166
|
+
previous: '‹ Prev',
|
|
167
|
+
next: 'Next ›',
|
|
168
|
+
last: 'Last »'
|
|
169
|
+
},
|
|
170
|
+
},
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
function renderDatetime(data, type, row) {
|
|
174
|
+
return (type === 'display') ? moment.utc(data).local().format('YYYY-MM-DD h:mm:ss a') : data;
|
|
175
|
+
}
|
|
176
|
+
function renderParameterButton(data, type, row) {
|
|
177
|
+
if (type == 'display') {
|
|
178
|
+
return (
|
|
179
|
+
`<button type="button" class="btn btn-link btn-sm"
|
|
180
|
+
data-bs-toggle="modal"
|
|
181
|
+
data-bs-target="#param-modal"
|
|
182
|
+
data-bs-param-name=${data}
|
|
183
|
+
data-bs-job-type=${row.job_type}
|
|
184
|
+
data-bs-task-id=${row.task_id}
|
|
185
|
+
data-bs-modality=${row.modality}
|
|
186
|
+
>${data}
|
|
187
|
+
</button>`
|
|
188
|
+
);
|
|
189
|
+
}
|
|
190
|
+
return data;
|
|
191
|
+
}
|
|
192
|
+
</script>
|
|
193
|
+
</body>
|
|
194
|
+
</html>
|