forked from umple/umple
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdockerExecution.js
121 lines (102 loc) · 4.5 KB
/
dockerExecution.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
const fs = require('fs');
const exec = require('child_process').exec;
class DockerExecution {
constructor(path, mainFile, model) {
this.path = path;
this.mainFile = mainFile;
this.model = model;
this.outputFolder = "output/" + this.model + "_" + this.mainFile;
const config = this.readConfig();
this.basePath = config['umplePath'];
this.baseOutputPath = config['tempPath'];
this.tempContainerName = config['tempContainerName'];
this.timeoutValue = +config['timeoutValue'];
}
run(callback) {
// Make output folder where the output files will be written
this.makeOutputFolder();
const mainFilePath = this.getNormalizedMainFilename();
console.log("Normalized main file: ", mainFilePath);
const command = `sh dockerTimeout.sh ${this.timeoutValue}s -i -t --network none -v ${this.basePath}/${this.model}:/input/:ro -v ${this.baseOutputPath}/${this.model}_${this.mainFile}:/output/ ${this.tempContainerName} ${mainFilePath}`
console.log("Docker command:'",command,"'");
exec(command); // Execute docker for Java execution
this.listenToChanges(callback);
}
getNormalizedMainFilename() {
let path = this.path;
if(path.startsWith('/')) {
path = path.substring(1);
}
if(path.endsWith('/')) {
path = path.substring(0, path.length - 1);
}
return path ? `${path.split('/').join('.')}.${this.mainFile}` : this.mainFile;
}
listenToChanges(callback) {
//variable to enforce the timeout_value
let timeValue = 0;
// Check For File named "completed" or "errors" after every 1 second
const intid = setInterval(() => {
timeValue++;
// Listen for the completed file
fs.readFile(this.outputFolder + '/completed', 'utf8', (err, completeData) => {
// if file is not available yet and the file interval is not yet up carry on
// else if file is found simply display a message and proceed
if(err && timeValue < this.timeoutValue) {
return;
} else if (timeValue < this.timeoutValue) {
//check for possible errors
fs.readFile(this.outputFolder + '/errors', 'utf8', (err, errorData) => {
if(errorData) {
console.log("Error file: ", errorData)
}
console.log("Complete file: ", completeData);
callback(errorData, completeData.toString())
});
} else {
// if time is up. Save an error message to the data variable
// Since the time is up, we take the partial output and return it.
fs.readFile(this.outputFolder + '/logfile.txt', 'utf8', (err, partialData) => {
if (!partialData) partialData = "";
partialData += "\nExecution Timed Out. Maximum allowed time is " + this.timeoutValue + " seconds.";
fs.readFile(this.outputFolder + '/errors', 'utf8', (err, errorData) => {
callback(errorData ,partialData.toString())
});
});
}
// If time is finished, remove directory and clear timer
this.deleteOutputFolder();
clearInterval(intid);
});
}, 1000);
}
makeOutputFolder() {
if(fs.existsSync(this.outputFolder)) {
this.deleteOutputFolder();
} else {
fs.mkdirSync(this.outputFolder);
}
}
deleteOutputFolder() {
try {
console.log("ATTEMPTING TO REMOVE: " + this.outputFolder);
fs.rmSync(this.outputFolder, { recursive: true });
console.log(`${this.outputFolder} is deleted!`);
} catch (err) {
console.error(`Error while deleting ${this.outputFolder}.`);
}
}
readConfig() {
const file = fs.readFileSync('./config.cfg', 'utf8');
const config = file.toString().replace(/\r\n/g,'\n').split('\n');
const obj = {};
for(let c of config) {
const cur = c.split('=');
obj[cur[0]] = cur[1];
}
console.log("Given Config:");
console.log(obj);
return obj;
}
}
module.exports = DockerExecution;