@sassoftware/viya-serverjs 0.2.4 → 0.4.0
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.
- package/.babelrc +6 -6
- package/.dockerignore +2 -2
- package/.env +29 -29
- package/.env.proxy +32 -32
- package/.env.server +30 -30
- package/.eslintignore +3 -3
- package/.eslintrc.json +42 -42
- package/Dockerfile +44 -44
- package/README.md +99 -99
- package/cli.js +9 -9
- package/lib/config.js +16 -16
- package/lib/handlers/appCallback.js +23 -24
- package/lib/handlers/codeAuth.js +17 -18
- package/lib/handlers/decodeJwt.js +3 -3
- package/lib/handlers/favicon.js +22 -25
- package/lib/handlers/getApp.js +20 -21
- package/lib/handlers/getApp2.js +22 -25
- package/lib/handlers/getUser.js +14 -17
- package/lib/handlers/index.js +3 -3
- package/lib/handlers/keepAlive.js +28 -31
- package/lib/handlers/keepAlive2.js +9 -12
- package/lib/handlers/logon.js +12 -15
- package/lib/handlers/logout.js +24 -28
- package/lib/handlers/proxyMapUri.js +6 -11
- package/lib/handlers/proxyOnResponse.js +6 -7
- package/lib/handlers/reactDev.js +22 -25
- package/lib/handlers/setCookies.js +52 -53
- package/lib/iService.js +106 -104
- package/lib/index.js +23 -21
- package/lib/parseDocker.js +3 -3
- package/lib/plugins/SASauth.js +30 -32
- package/lib/plugins/appCookie.js +36 -38
- package/lib/plugins/setContext.js +22 -25
- package/lib/plugins/setDefaultRoutes.js +61 -58
- package/lib/plugins/setupAuth.js +35 -38
- package/lib/plugins/setupUserRoutes.js +16 -16
- package/lib/plugins/token.js +9 -10
- package/lib/schemes/SASTokenScheme.js +24 -27
- package/package.json +85 -79
- package/public/data/Cluster_SDOH1.sas +181 -181
- package/public/data/Cluster_SDOH6.sas +179 -179
- package/public/data/NeuralNetwork_High_med.sas +2408 -2408
- package/public/data/NeuralNetwork_high_med1.sas +2408 -2408
- package/public/data/cars.csv +429 -429
- package/public/data/cmdList.txt +15 -15
- package/public/data/iris.csv +151 -151
- package/public/index.html +355 -360
- package/public/indexProxy.html +351 -351
- package/public/simplesubmit.html +233 -0
- package/server.js +362 -362
- package/src/config.js +90 -90
- package/src/handlers/appCallback.js +40 -40
- package/src/handlers/codeAuth.js +31 -31
- package/src/handlers/decodeJwt.js +10 -10
- package/src/handlers/favicon.js +23 -23
- package/src/handlers/getApp.js +52 -52
- package/src/handlers/getApp2.js +25 -25
- package/src/handlers/getUser.js +20 -20
- package/src/handlers/index.js +36 -36
- package/src/handlers/keepAlive.js +53 -53
- package/src/handlers/keepAlive2.js +12 -12
- package/src/handlers/logon.js +23 -23
- package/src/handlers/logout.js +42 -42
- package/src/handlers/proxyMapUri.js +25 -25
- package/src/handlers/proxyOnResponse.js +11 -11
- package/src/handlers/reactDev.js +29 -29
- package/src/handlers/setCookies.js +81 -79
- package/src/iService.js +346 -347
- package/src/index.js +251 -249
- package/src/parseDocker.js +32 -32
- package/src/plugins/SASauth.js +78 -78
- package/src/plugins/appCookie.js +51 -49
- package/src/plugins/setContext.js +33 -33
- package/src/plugins/setDefaultRoutes.js +256 -253
- package/src/plugins/setupAuth.js +49 -49
- package/src/plugins/setupUserRoutes.js +47 -47
- package/src/plugins/token.js +10 -10
- package/src/schemes/SASTokenScheme.js +43 -43
- package/src/visionIndex.html +23 -23
- package/start.sh +14 -14
- package/tls/viyatls.sh +2 -2
package/server.js
CHANGED
|
@@ -1,363 +1,363 @@
|
|
|
1
|
-
/*
|
|
2
|
-
* Copyright © 2025, SAS Institute Inc., Cary, NC, USA. All Rights Reserved.
|
|
3
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
4
|
-
*/
|
|
5
|
-
let core = require('./lib/index.js');
|
|
6
|
-
debugger;
|
|
7
|
-
core(getCustomHandler, true, 'app', null);
|
|
8
|
-
|
|
9
|
-
function getCustomHandler() {
|
|
10
|
-
let appName = `/${process.env.APPNAME}`; /* does not have to be this - your choice */
|
|
11
|
-
debugger;
|
|
12
|
-
let routes = [
|
|
13
|
-
{
|
|
14
|
-
method: ["GET"],
|
|
15
|
-
path: "/help",
|
|
16
|
-
options: {
|
|
17
|
-
files: {
|
|
18
|
-
relativeTo: "./public",
|
|
19
|
-
},
|
|
20
|
-
handler: async (req, h) => {
|
|
21
|
-
debugger;
|
|
22
|
-
let hf = 'help.html';
|
|
23
|
-
return h.file(hf);
|
|
24
|
-
},
|
|
25
|
-
auth: false,
|
|
26
|
-
description: "Help",
|
|
27
|
-
notes: "Help",
|
|
28
|
-
tags: ["app"],
|
|
29
|
-
},
|
|
30
|
-
},
|
|
31
|
-
{
|
|
32
|
-
method: ["GET"],
|
|
33
|
-
path: `${appName}/new`,
|
|
34
|
-
options: {
|
|
35
|
-
files: {
|
|
36
|
-
relativeTo: "./public",
|
|
37
|
-
},
|
|
38
|
-
handler: async (req, h) => {
|
|
39
|
-
console.log('>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>in new');
|
|
40
|
-
return h.file('index.html');
|
|
41
|
-
},
|
|
42
|
-
// auth: 'logon',
|
|
43
|
-
description: "Create new application",
|
|
44
|
-
notes: "Index file created from env data",
|
|
45
|
-
tags: ["app"],
|
|
46
|
-
},
|
|
47
|
-
}
|
|
48
|
-
];
|
|
49
|
-
return routes;
|
|
50
|
-
}
|
|
51
|
-
function customize(key, options) {
|
|
52
|
-
let info = {
|
|
53
|
-
swaggerOptions: {
|
|
54
|
-
info: {
|
|
55
|
-
title: "Test API",
|
|
56
|
-
version: "0.0.1",
|
|
57
|
-
description: "This document was auto-generated at run time",
|
|
58
|
-
},
|
|
59
|
-
documentationPage: true,
|
|
60
|
-
documentationPath: `/${process.env.APPNAME}/documentation`,
|
|
61
|
-
swaggerUI: true,
|
|
62
|
-
swaggerUIPath: `/${process.env.APPNAME}/swaggerui`,
|
|
63
|
-
schemes: ["https", "http"],
|
|
64
|
-
cors: true,
|
|
65
|
-
auth: options.authDefault,
|
|
66
|
-
},
|
|
67
|
-
APPENV: {
|
|
68
|
-
x: 1,
|
|
69
|
-
y: 2,
|
|
70
|
-
},
|
|
71
|
-
};
|
|
72
|
-
let r = info[key];
|
|
73
|
-
return r == null ? {} : r;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
function getIndex() {
|
|
77
|
-
|
|
78
|
-
let template = `
|
|
79
|
-
<html lang="en">
|
|
80
|
-
<head>
|
|
81
|
-
<meta charset="UTF-8" />
|
|
82
|
-
|
|
83
|
-
<script
|
|
84
|
-
crossorigin
|
|
85
|
-
src="https://unpkg.com/react@16/umd/react.production.min.js"
|
|
86
|
-
></script>
|
|
87
|
-
<script
|
|
88
|
-
crossorigin
|
|
89
|
-
src="https://unpkg.com/react-dom@16/umd/react-dom.production.min.js"
|
|
90
|
-
></script>
|
|
91
|
-
<script src="https://cdnjs.cloudflare.com/ajax/libs/babel-standalone/6.26.0/babel.min.js"></script>
|
|
92
|
-
<script src="https://cdnjs.cloudflare.com/ajax/libs/babel-polyfill/6.26.0/polyfill.min.js"></script>
|
|
93
|
-
<script src="https://unpkg.com/@sassoftware/restaf@5.2.4/dist/restaf.js"></script>
|
|
94
|
-
<script src="https://unpkg.com/@sassoftware/restaflib@5.2.4/dist/restaflib.js"></script>
|
|
95
|
-
|
|
96
|
-
<style>
|
|
97
|
-
.container {
|
|
98
|
-
display: flex;
|
|
99
|
-
flex-direction: column;
|
|
100
|
-
flex-wrap: nowrap;
|
|
101
|
-
min-height: 800px;
|
|
102
|
-
}
|
|
103
|
-
.elabel {
|
|
104
|
-
display: inline-block;
|
|
105
|
-
|
|
106
|
-
clear: left;
|
|
107
|
-
width: 250px;
|
|
108
|
-
text-align: right;
|
|
109
|
-
}
|
|
110
|
-
.einput {
|
|
111
|
-
display: inline-block;
|
|
112
|
-
}
|
|
113
|
-
.div1 {
|
|
114
|
-
border: 1px solid black;
|
|
115
|
-
background: lightskyblue;
|
|
116
|
-
}
|
|
117
|
-
.div2 {
|
|
118
|
-
border: 1px solid black;
|
|
119
|
-
background: lightskyblue;
|
|
120
|
-
height: 200px;
|
|
121
|
-
}
|
|
122
|
-
</style>
|
|
123
|
-
|
|
124
|
-
<script>
|
|
125
|
-
let LOGONPAYLOAD = {
|
|
126
|
-
host: "${process.env.VIYA_SERVER}",
|
|
127
|
-
authType: "server",
|
|
128
|
-
appName: "${process.env.APPNAME}",
|
|
129
|
-
};
|
|
130
|
-
</script>
|
|
131
|
-
|
|
132
|
-
<script>
|
|
133
|
-
debugger;
|
|
134
|
-
let store = restaf.initStore({
|
|
135
|
-
casProxy: true});
|
|
136
|
-
debugger; console.log(store.config);
|
|
137
|
-
|
|
138
|
-
let session = null;
|
|
139
|
-
let servers = null;
|
|
140
|
-
let services = null;
|
|
141
|
-
let files = null;
|
|
142
|
-
let reports = null;
|
|
143
|
-
let compute = null;
|
|
144
|
-
|
|
145
|
-
function setup() {
|
|
146
|
-
debugger;
|
|
147
|
-
document.getElementById('output').innerHTML = '...initializing';
|
|
148
|
-
|
|
149
|
-
initSession()
|
|
150
|
-
.then(r => {
|
|
151
|
-
document.getElementById('output').innerHTML = 'ready';
|
|
152
|
-
keepAlive();
|
|
153
|
-
})
|
|
154
|
-
.catch(e => {
|
|
155
|
-
|
|
156
|
-
console.log(e);
|
|
157
|
-
});
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
async function initSession() {
|
|
162
|
-
|
|
163
|
-
console.log(LOGONPAYLOAD);
|
|
164
|
-
debugger;
|
|
165
|
-
let msg = await store.logon(LOGONPAYLOAD);
|
|
166
|
-
console.log(msg);
|
|
167
|
-
|
|
168
|
-
// let { identities } = await store.addServices('identities');
|
|
169
|
-
let name = 'user';
|
|
170
|
-
// if (identities.links('currentUser') != null) {
|
|
171
|
-
// let c = await store.apiCall(identities.links('currentUser'));
|
|
172
|
-
// name = c.items('id');
|
|
173
|
-
// }
|
|
174
|
-
console.log(name);
|
|
175
|
-
debugger;
|
|
176
|
-
/*
|
|
177
|
-
services = await store.addServices(
|
|
178
|
-
'files', 'compute', 'casManagement'
|
|
179
|
-
);
|
|
180
|
-
console.log(services.casManagement.links().toJS())
|
|
181
|
-
*/
|
|
182
|
-
debugger;
|
|
183
|
-
return 'done';
|
|
184
|
-
}
|
|
185
|
-
function runit(type) {
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
document.getElementById('output').innerHTML = '...running';
|
|
189
|
-
let testcase;
|
|
190
|
-
switch (type) {
|
|
191
|
-
case 'files': {
|
|
192
|
-
testcase = SASfileService;
|
|
193
|
-
break;
|
|
194
|
-
}
|
|
195
|
-
case 'compute': {
|
|
196
|
-
testcase = dsCompute;
|
|
197
|
-
break;
|
|
198
|
-
}
|
|
199
|
-
case 'cas': {
|
|
200
|
-
testcase = runCas;
|
|
201
|
-
break;
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
case 'spre': {
|
|
205
|
-
testcase= spre;
|
|
206
|
-
|
|
207
|
-
break;
|
|
208
|
-
}
|
|
209
|
-
default: {
|
|
210
|
-
testcase = SASfileService;
|
|
211
|
-
break;
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
testcase(store)
|
|
216
|
-
.then(r => {
|
|
217
|
-
document.getElementById(
|
|
218
|
-
'output'
|
|
219
|
-
).innerHTML = JSON.stringify(r, null, 4);
|
|
220
|
-
})
|
|
221
|
-
.catch(err => {
|
|
222
|
-
|
|
223
|
-
document.getElementById(
|
|
224
|
-
'output'
|
|
225
|
-
).innerHTML = JSON.stringify(err, null, 4);
|
|
226
|
-
});
|
|
227
|
-
}
|
|
228
|
-
async function noaction() {
|
|
229
|
-
r = {msg: 'redirects completed'};
|
|
230
|
-
return r;
|
|
231
|
-
}
|
|
232
|
-
async function spre(store) {
|
|
233
|
-
let p = {
|
|
234
|
-
method: 'GET',
|
|
235
|
-
url : 'http://localhost:3000/api/test',
|
|
236
|
-
withCredentials: true
|
|
237
|
-
}
|
|
238
|
-
let r = await store.request(p);
|
|
239
|
-
return r.data;
|
|
240
|
-
}
|
|
241
|
-
async function runCas(store) {
|
|
242
|
-
|
|
243
|
-
let {casManagement} = await store.addServices('casManagement');
|
|
244
|
-
let servers = await store.apiCall(
|
|
245
|
-
casManagement.links('servers')
|
|
246
|
-
);
|
|
247
|
-
let serverName = servers.itemsList(0);
|
|
248
|
-
let session = await store.apiCall(
|
|
249
|
-
servers.itemsCmd(serverName, 'createSession')
|
|
250
|
-
);
|
|
251
|
-
let payload = {
|
|
252
|
-
action: 'builtins.echo',
|
|
253
|
-
data: { code: { x: 1 } }
|
|
254
|
-
};
|
|
255
|
-
console.log(JSON.stringify(session.links("execute"), null, 4));
|
|
256
|
-
let r = await store.runAction(session, payload);
|
|
257
|
-
console.log('echo completed');
|
|
258
|
-
await store.apiCall(session.links('delete'));
|
|
259
|
-
return r.items();
|
|
260
|
-
}
|
|
261
|
-
|
|
262
|
-
async function SASfileService(store) {
|
|
263
|
-
debugger;
|
|
264
|
-
let content;
|
|
265
|
-
try {
|
|
266
|
-
debugger;
|
|
267
|
-
let {files} = await store.addServices('files');
|
|
268
|
-
debugger;
|
|
269
|
-
console.log(JSON.stringify(files.links(), null, 4));
|
|
270
|
-
//console.log('items - should be an array of files(empty array is ok)')
|
|
271
|
-
// console.log(files.items().toJS());
|
|
272
|
-
let payload = {
|
|
273
|
-
data: { x: 1, y: 'This was saved earlier in the step' },
|
|
274
|
-
headers: { 'content-type': 'application/json' }
|
|
275
|
-
};
|
|
276
|
-
let createCmd = files.links('create');
|
|
277
|
-
let newFile = await store.apiCall(createCmd, payload);
|
|
278
|
-
debugger;
|
|
279
|
-
console.log(JSON.stringify(newFile.links('content'), null, 4));
|
|
280
|
-
content = await store.apiCall(newFile.links('content'));
|
|
281
|
-
|
|
282
|
-
} catch(err) {
|
|
283
|
-
console.log(JSON.stringify(err, null, 4));
|
|
284
|
-
debugger;
|
|
285
|
-
}
|
|
286
|
-
console.log(content);
|
|
287
|
-
return content.items();
|
|
288
|
-
}
|
|
289
|
-
async function dsCompute(store) {
|
|
290
|
-
let log = null;
|
|
291
|
-
debugger;
|
|
292
|
-
let {compute} = await store.addServices('compute');
|
|
293
|
-
let servers = await store.apiCall(compute.links('servers'));
|
|
294
|
-
|
|
295
|
-
let contexts = await store.apiCall(compute.links('contexts'));
|
|
296
|
-
|
|
297
|
-
// lookup the name of the first context and then use it to get the associated createSession restafLink
|
|
298
|
-
let createSession = contexts.itemsCmd(
|
|
299
|
-
contexts.itemsList(0),
|
|
300
|
-
'createSession'
|
|
301
|
-
);
|
|
302
|
-
let session = await store.apiCall(createSession);
|
|
303
|
-
|
|
304
|
-
// Now run a simple data step in that session
|
|
305
|
-
let payload = {
|
|
306
|
-
data: {
|
|
307
|
-
code: ["data _null_; do i = 1 to 100; x=1; end; run; "]
|
|
308
|
-
}
|
|
309
|
-
};
|
|
310
|
-
|
|
311
|
-
// Now execute the data step and wait for completion
|
|
312
|
-
let job = await store.apiCall(
|
|
313
|
-
session.links('execute'),
|
|
314
|
-
payload
|
|
315
|
-
);
|
|
316
|
-
let status = await store.jobState(job, null, 5, 2);
|
|
317
|
-
|
|
318
|
-
if (status.data === 'running') {
|
|
319
|
-
throw "ERROR: Job did not complete in allotted time";
|
|
320
|
-
} else {
|
|
321
|
-
switch (status.data) {
|
|
322
|
-
case 'warning':
|
|
323
|
-
console.log("Warning: check your log for warnings");
|
|
324
|
-
break;
|
|
325
|
-
case 'error':
|
|
326
|
-
throw "Please correct errors and rerun program";
|
|
327
|
-
default:
|
|
328
|
-
log = await store.apiCall(status.job.links('log'));
|
|
329
|
-
break;
|
|
330
|
-
}
|
|
331
|
-
}
|
|
332
|
-
return log === null ? status : log.items();
|
|
333
|
-
}
|
|
334
|
-
</script>
|
|
335
|
-
</head>
|
|
336
|
-
<body onload="setup()">
|
|
337
|
-
<h1 id="head">Hi</h1>
|
|
338
|
-
<div>
|
|
339
|
-
|
|
340
|
-
<button onclick="runit('files')">
|
|
341
|
-
Press to make a call to file service
|
|
342
|
-
</button>
|
|
343
|
-
<br />
|
|
344
|
-
<br />
|
|
345
|
-
<button onclick="runit('compute')">
|
|
346
|
-
Press to make a call compute service
|
|
347
|
-
</button>
|
|
348
|
-
<br />
|
|
349
|
-
<br />
|
|
350
|
-
<button onclick="runit('cas')">
|
|
351
|
-
Press to make a call to cas echo
|
|
352
|
-
</button>
|
|
353
|
-
<br />
|
|
354
|
-
<br />
|
|
355
|
-
|
|
356
|
-
<div>
|
|
357
|
-
<pre id="output"></pre>
|
|
358
|
-
</div>
|
|
359
|
-
</body>
|
|
360
|
-
</html>
|
|
361
|
-
`;
|
|
362
|
-
return template;
|
|
1
|
+
/*
|
|
2
|
+
* Copyright © 2025, SAS Institute Inc., Cary, NC, USA. All Rights Reserved.
|
|
3
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
*/
|
|
5
|
+
let core = require('./lib/index.js');
|
|
6
|
+
debugger;
|
|
7
|
+
core(getCustomHandler, true, 'app', null);
|
|
8
|
+
|
|
9
|
+
function getCustomHandler() {
|
|
10
|
+
let appName = `/${process.env.APPNAME}`; /* does not have to be this - your choice */
|
|
11
|
+
debugger;
|
|
12
|
+
let routes = [
|
|
13
|
+
{
|
|
14
|
+
method: ["GET"],
|
|
15
|
+
path: "/help",
|
|
16
|
+
options: {
|
|
17
|
+
files: {
|
|
18
|
+
relativeTo: "./public",
|
|
19
|
+
},
|
|
20
|
+
handler: async (req, h) => {
|
|
21
|
+
debugger;
|
|
22
|
+
let hf = 'help.html';
|
|
23
|
+
return h.file(hf);
|
|
24
|
+
},
|
|
25
|
+
auth: false,
|
|
26
|
+
description: "Help",
|
|
27
|
+
notes: "Help",
|
|
28
|
+
tags: ["app"],
|
|
29
|
+
},
|
|
30
|
+
},
|
|
31
|
+
{
|
|
32
|
+
method: ["GET"],
|
|
33
|
+
path: `${appName}/new`,
|
|
34
|
+
options: {
|
|
35
|
+
files: {
|
|
36
|
+
relativeTo: "./public",
|
|
37
|
+
},
|
|
38
|
+
handler: async (req, h) => {
|
|
39
|
+
console.log('>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>in new');
|
|
40
|
+
return h.file('index.html');
|
|
41
|
+
},
|
|
42
|
+
// auth: 'logon',
|
|
43
|
+
description: "Create new application",
|
|
44
|
+
notes: "Index file created from env data",
|
|
45
|
+
tags: ["app"],
|
|
46
|
+
},
|
|
47
|
+
}
|
|
48
|
+
];
|
|
49
|
+
return routes;
|
|
50
|
+
}
|
|
51
|
+
function customize(key, options) {
|
|
52
|
+
let info = {
|
|
53
|
+
swaggerOptions: {
|
|
54
|
+
info: {
|
|
55
|
+
title: "Test API",
|
|
56
|
+
version: "0.0.1",
|
|
57
|
+
description: "This document was auto-generated at run time",
|
|
58
|
+
},
|
|
59
|
+
documentationPage: true,
|
|
60
|
+
documentationPath: `/${process.env.APPNAME}/documentation`,
|
|
61
|
+
swaggerUI: true,
|
|
62
|
+
swaggerUIPath: `/${process.env.APPNAME}/swaggerui`,
|
|
63
|
+
schemes: ["https", "http"],
|
|
64
|
+
cors: true,
|
|
65
|
+
auth: options.authDefault,
|
|
66
|
+
},
|
|
67
|
+
APPENV: {
|
|
68
|
+
x: 1,
|
|
69
|
+
y: 2,
|
|
70
|
+
},
|
|
71
|
+
};
|
|
72
|
+
let r = info[key];
|
|
73
|
+
return r == null ? {} : r;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function getIndex() {
|
|
77
|
+
|
|
78
|
+
let template = `
|
|
79
|
+
<html lang="en">
|
|
80
|
+
<head>
|
|
81
|
+
<meta charset="UTF-8" />
|
|
82
|
+
|
|
83
|
+
<script
|
|
84
|
+
crossorigin
|
|
85
|
+
src="https://unpkg.com/react@16/umd/react.production.min.js"
|
|
86
|
+
></script>
|
|
87
|
+
<script
|
|
88
|
+
crossorigin
|
|
89
|
+
src="https://unpkg.com/react-dom@16/umd/react-dom.production.min.js"
|
|
90
|
+
></script>
|
|
91
|
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/babel-standalone/6.26.0/babel.min.js"></script>
|
|
92
|
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/babel-polyfill/6.26.0/polyfill.min.js"></script>
|
|
93
|
+
<script src="https://unpkg.com/@sassoftware/restaf@5.2.4/dist/restaf.js"></script>
|
|
94
|
+
<script src="https://unpkg.com/@sassoftware/restaflib@5.2.4/dist/restaflib.js"></script>
|
|
95
|
+
|
|
96
|
+
<style>
|
|
97
|
+
.container {
|
|
98
|
+
display: flex;
|
|
99
|
+
flex-direction: column;
|
|
100
|
+
flex-wrap: nowrap;
|
|
101
|
+
min-height: 800px;
|
|
102
|
+
}
|
|
103
|
+
.elabel {
|
|
104
|
+
display: inline-block;
|
|
105
|
+
|
|
106
|
+
clear: left;
|
|
107
|
+
width: 250px;
|
|
108
|
+
text-align: right;
|
|
109
|
+
}
|
|
110
|
+
.einput {
|
|
111
|
+
display: inline-block;
|
|
112
|
+
}
|
|
113
|
+
.div1 {
|
|
114
|
+
border: 1px solid black;
|
|
115
|
+
background: lightskyblue;
|
|
116
|
+
}
|
|
117
|
+
.div2 {
|
|
118
|
+
border: 1px solid black;
|
|
119
|
+
background: lightskyblue;
|
|
120
|
+
height: 200px;
|
|
121
|
+
}
|
|
122
|
+
</style>
|
|
123
|
+
|
|
124
|
+
<script>
|
|
125
|
+
let LOGONPAYLOAD = {
|
|
126
|
+
host: "${process.env.VIYA_SERVER}",
|
|
127
|
+
authType: "server",
|
|
128
|
+
appName: "${process.env.APPNAME}",
|
|
129
|
+
};
|
|
130
|
+
</script>
|
|
131
|
+
|
|
132
|
+
<script>
|
|
133
|
+
debugger;
|
|
134
|
+
let store = restaf.initStore({
|
|
135
|
+
casProxy: true});
|
|
136
|
+
debugger; console.log(store.config);
|
|
137
|
+
|
|
138
|
+
let session = null;
|
|
139
|
+
let servers = null;
|
|
140
|
+
let services = null;
|
|
141
|
+
let files = null;
|
|
142
|
+
let reports = null;
|
|
143
|
+
let compute = null;
|
|
144
|
+
|
|
145
|
+
function setup() {
|
|
146
|
+
debugger;
|
|
147
|
+
document.getElementById('output').innerHTML = '...initializing';
|
|
148
|
+
|
|
149
|
+
initSession()
|
|
150
|
+
.then(r => {
|
|
151
|
+
document.getElementById('output').innerHTML = 'ready';
|
|
152
|
+
keepAlive();
|
|
153
|
+
})
|
|
154
|
+
.catch(e => {
|
|
155
|
+
|
|
156
|
+
console.log(e);
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
|
|
161
|
+
async function initSession() {
|
|
162
|
+
|
|
163
|
+
console.log(LOGONPAYLOAD);
|
|
164
|
+
debugger;
|
|
165
|
+
let msg = await store.logon(LOGONPAYLOAD);
|
|
166
|
+
console.log(msg);
|
|
167
|
+
|
|
168
|
+
// let { identities } = await store.addServices('identities');
|
|
169
|
+
let name = 'user';
|
|
170
|
+
// if (identities.links('currentUser') != null) {
|
|
171
|
+
// let c = await store.apiCall(identities.links('currentUser'));
|
|
172
|
+
// name = c.items('id');
|
|
173
|
+
// }
|
|
174
|
+
console.log(name);
|
|
175
|
+
debugger;
|
|
176
|
+
/*
|
|
177
|
+
services = await store.addServices(
|
|
178
|
+
'files', 'compute', 'casManagement'
|
|
179
|
+
);
|
|
180
|
+
console.log(services.casManagement.links().toJS())
|
|
181
|
+
*/
|
|
182
|
+
debugger;
|
|
183
|
+
return 'done';
|
|
184
|
+
}
|
|
185
|
+
function runit(type) {
|
|
186
|
+
|
|
187
|
+
|
|
188
|
+
document.getElementById('output').innerHTML = '...running';
|
|
189
|
+
let testcase;
|
|
190
|
+
switch (type) {
|
|
191
|
+
case 'files': {
|
|
192
|
+
testcase = SASfileService;
|
|
193
|
+
break;
|
|
194
|
+
}
|
|
195
|
+
case 'compute': {
|
|
196
|
+
testcase = dsCompute;
|
|
197
|
+
break;
|
|
198
|
+
}
|
|
199
|
+
case 'cas': {
|
|
200
|
+
testcase = runCas;
|
|
201
|
+
break;
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
case 'spre': {
|
|
205
|
+
testcase= spre;
|
|
206
|
+
|
|
207
|
+
break;
|
|
208
|
+
}
|
|
209
|
+
default: {
|
|
210
|
+
testcase = SASfileService;
|
|
211
|
+
break;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
testcase(store)
|
|
216
|
+
.then(r => {
|
|
217
|
+
document.getElementById(
|
|
218
|
+
'output'
|
|
219
|
+
).innerHTML = JSON.stringify(r, null, 4);
|
|
220
|
+
})
|
|
221
|
+
.catch(err => {
|
|
222
|
+
|
|
223
|
+
document.getElementById(
|
|
224
|
+
'output'
|
|
225
|
+
).innerHTML = JSON.stringify(err, null, 4);
|
|
226
|
+
});
|
|
227
|
+
}
|
|
228
|
+
async function noaction() {
|
|
229
|
+
r = {msg: 'redirects completed'};
|
|
230
|
+
return r;
|
|
231
|
+
}
|
|
232
|
+
async function spre(store) {
|
|
233
|
+
let p = {
|
|
234
|
+
method: 'GET',
|
|
235
|
+
url : 'http://localhost:3000/api/test',
|
|
236
|
+
withCredentials: true
|
|
237
|
+
}
|
|
238
|
+
let r = await store.request(p);
|
|
239
|
+
return r.data;
|
|
240
|
+
}
|
|
241
|
+
async function runCas(store) {
|
|
242
|
+
|
|
243
|
+
let {casManagement} = await store.addServices('casManagement');
|
|
244
|
+
let servers = await store.apiCall(
|
|
245
|
+
casManagement.links('servers')
|
|
246
|
+
);
|
|
247
|
+
let serverName = servers.itemsList(0);
|
|
248
|
+
let session = await store.apiCall(
|
|
249
|
+
servers.itemsCmd(serverName, 'createSession')
|
|
250
|
+
);
|
|
251
|
+
let payload = {
|
|
252
|
+
action: 'builtins.echo',
|
|
253
|
+
data: { code: { x: 1 } }
|
|
254
|
+
};
|
|
255
|
+
console.log(JSON.stringify(session.links("execute"), null, 4));
|
|
256
|
+
let r = await store.runAction(session, payload);
|
|
257
|
+
console.log('echo completed');
|
|
258
|
+
await store.apiCall(session.links('delete'));
|
|
259
|
+
return r.items();
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
async function SASfileService(store) {
|
|
263
|
+
debugger;
|
|
264
|
+
let content;
|
|
265
|
+
try {
|
|
266
|
+
debugger;
|
|
267
|
+
let {files} = await store.addServices('files');
|
|
268
|
+
debugger;
|
|
269
|
+
console.log(JSON.stringify(files.links(), null, 4));
|
|
270
|
+
//console.log('items - should be an array of files(empty array is ok)')
|
|
271
|
+
// console.log(files.items().toJS());
|
|
272
|
+
let payload = {
|
|
273
|
+
data: { x: 1, y: 'This was saved earlier in the step' },
|
|
274
|
+
headers: { 'content-type': 'application/json' }
|
|
275
|
+
};
|
|
276
|
+
let createCmd = files.links('create');
|
|
277
|
+
let newFile = await store.apiCall(createCmd, payload);
|
|
278
|
+
debugger;
|
|
279
|
+
console.log(JSON.stringify(newFile.links('content'), null, 4));
|
|
280
|
+
content = await store.apiCall(newFile.links('content'));
|
|
281
|
+
|
|
282
|
+
} catch(err) {
|
|
283
|
+
console.log(JSON.stringify(err, null, 4));
|
|
284
|
+
debugger;
|
|
285
|
+
}
|
|
286
|
+
console.log(content);
|
|
287
|
+
return content.items();
|
|
288
|
+
}
|
|
289
|
+
async function dsCompute(store) {
|
|
290
|
+
let log = null;
|
|
291
|
+
debugger;
|
|
292
|
+
let {compute} = await store.addServices('compute');
|
|
293
|
+
let servers = await store.apiCall(compute.links('servers'));
|
|
294
|
+
|
|
295
|
+
let contexts = await store.apiCall(compute.links('contexts'));
|
|
296
|
+
|
|
297
|
+
// lookup the name of the first context and then use it to get the associated createSession restafLink
|
|
298
|
+
let createSession = contexts.itemsCmd(
|
|
299
|
+
contexts.itemsList(0),
|
|
300
|
+
'createSession'
|
|
301
|
+
);
|
|
302
|
+
let session = await store.apiCall(createSession);
|
|
303
|
+
|
|
304
|
+
// Now run a simple data step in that session
|
|
305
|
+
let payload = {
|
|
306
|
+
data: {
|
|
307
|
+
code: ["data _null_; do i = 1 to 100; x=1; end; run; "]
|
|
308
|
+
}
|
|
309
|
+
};
|
|
310
|
+
|
|
311
|
+
// Now execute the data step and wait for completion
|
|
312
|
+
let job = await store.apiCall(
|
|
313
|
+
session.links('execute'),
|
|
314
|
+
payload
|
|
315
|
+
);
|
|
316
|
+
let status = await store.jobState(job, null, 5, 2);
|
|
317
|
+
|
|
318
|
+
if (status.data === 'running') {
|
|
319
|
+
throw "ERROR: Job did not complete in allotted time";
|
|
320
|
+
} else {
|
|
321
|
+
switch (status.data) {
|
|
322
|
+
case 'warning':
|
|
323
|
+
console.log("Warning: check your log for warnings");
|
|
324
|
+
break;
|
|
325
|
+
case 'error':
|
|
326
|
+
throw "Please correct errors and rerun program";
|
|
327
|
+
default:
|
|
328
|
+
log = await store.apiCall(status.job.links('log'));
|
|
329
|
+
break;
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
return log === null ? status : log.items();
|
|
333
|
+
}
|
|
334
|
+
</script>
|
|
335
|
+
</head>
|
|
336
|
+
<body onload="setup()">
|
|
337
|
+
<h1 id="head">Hi</h1>
|
|
338
|
+
<div>
|
|
339
|
+
|
|
340
|
+
<button onclick="runit('files')">
|
|
341
|
+
Press to make a call to file service
|
|
342
|
+
</button>
|
|
343
|
+
<br />
|
|
344
|
+
<br />
|
|
345
|
+
<button onclick="runit('compute')">
|
|
346
|
+
Press to make a call compute service
|
|
347
|
+
</button>
|
|
348
|
+
<br />
|
|
349
|
+
<br />
|
|
350
|
+
<button onclick="runit('cas')">
|
|
351
|
+
Press to make a call to cas echo
|
|
352
|
+
</button>
|
|
353
|
+
<br />
|
|
354
|
+
<br />
|
|
355
|
+
|
|
356
|
+
<div>
|
|
357
|
+
<pre id="output"></pre>
|
|
358
|
+
</div>
|
|
359
|
+
</body>
|
|
360
|
+
</html>
|
|
361
|
+
`;
|
|
362
|
+
return template;
|
|
363
363
|
}
|