├── README.md ├── app.js ├── bin └── www ├── docker ├── javascript │ └── Dockerfile ├── php │ └── Dockerfile ├── readme.md └── script.sh ├── package-lock.json ├── package.json ├── public └── stylesheets │ └── style.css ├── routes └── index.js └── views ├── error.jade ├── index.jade └── layout.jade /README.md: -------------------------------------------------------------------------------- 1 | # Code challenge by docker sandbox 2 | 3 | ## Build image cho docker 4 | 5 | ### PHP 6 | 7 | ``` 8 | cd docker 9 | docker build -t ntcd_php -f php/Dockerfile . 10 | docker run -it --rm -v /tmp:/tmp ntcd_php php /tmp/main.php /tmp/output 11 | ``` 12 | 13 | ### Javascript 14 | 15 | ``` 16 | cd docker 17 | docker build -t ntcd_javascript -f javascript/Dockerfile . 18 | docker run -it --rm -v /tmp:/tmp ntcd_javascript node /tmp/main.js /tmp/output 19 | ``` 20 | 21 | ## Start API 22 | 23 | ``` 24 | npm run start 25 | ``` 26 | 27 | ## API 28 | 29 | Endpoint: http://localhost:3000/run/:language 30 | 31 | Request 32 | 33 | ``` 34 | { 35 | "content": "= 0) { 45 | // port number 46 | return port; 47 | } 48 | 49 | return false; 50 | } 51 | 52 | /** 53 | * Event listener for HTTP server "error" event. 54 | */ 55 | 56 | function onError(error) { 57 | if (error.syscall !== 'listen') { 58 | throw error; 59 | } 60 | 61 | var bind = typeof port === 'string' 62 | ? 'Pipe ' + port 63 | : 'Port ' + port; 64 | 65 | // handle specific listen errors with friendly messages 66 | switch (error.code) { 67 | case 'EACCES': 68 | console.error(bind + ' requires elevated privileges'); 69 | process.exit(1); 70 | break; 71 | case 'EADDRINUSE': 72 | console.error(bind + ' is already in use'); 73 | process.exit(1); 74 | break; 75 | default: 76 | throw error; 77 | } 78 | } 79 | 80 | /** 81 | * Event listener for HTTP server "listening" event. 82 | */ 83 | 84 | function onListening() { 85 | var addr = server.address(); 86 | var bind = typeof addr === 'string' 87 | ? 'pipe ' + addr 88 | : 'port ' + addr.port; 89 | debug('Listening on ' + bind); 90 | } 91 | -------------------------------------------------------------------------------- /docker/javascript/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM node:8.16.0-alpine 2 | 3 | RUN apk add --no-cache bc 4 | RUN apk add --update coreutils && rm -rf /var/cache/apk/* 5 | 6 | WORKDIR /home/ntcd/ 7 | 8 | ADD script.sh /bin/script.sh 9 | RUN chmod +x /bin/script.sh 10 | 11 | ENTRYPOINT ["sh", "/bin/script.sh"] -------------------------------------------------------------------------------- /docker/php/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM php:7.3.6-cli-alpine3.9 2 | 3 | RUN apk add --no-cache bc 4 | RUN apk add --update coreutils && rm -rf /var/cache/apk/* 5 | 6 | WORKDIR /home/ntcd/ 7 | 8 | ADD script.sh /bin/script.sh 9 | RUN chmod +x /bin/script.sh 10 | 11 | ENTRYPOINT ["sh", "/bin/script.sh"] -------------------------------------------------------------------------------- /docker/readme.md: -------------------------------------------------------------------------------- 1 | ### PHP 2 | 3 | ``` 4 | docker build -t ntcd_php -f php/Dockerfile . 5 | docker run -it --rm -v /tmp:/tmp ntcd_php php /tmp/main.php /tmp/output 6 | ``` 7 | 8 | ### Javascript 9 | 10 | ``` 11 | docker build -t ntcd_javascript -f javascript/Dockerfile . 12 | docker run -it --rm -v /tmp:/tmp ntcd_javascript node /tmp/main.js /tmp/output 13 | ``` -------------------------------------------------------------------------------- /docker/script.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | # Sample execution command: $: ./script.sh php main.php 4 | 5 | compiler=$1 6 | file=$2 7 | output=$3 8 | 9 | START=$(date +%s.%4N) 10 | 11 | $compiler $file > $output 12 | 13 | END=$(date +%s.%4N) 14 | 15 | runtime=$(echo "$END - $START" | bc) 16 | 17 | echo $runtime 18 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "express-example", 3 | "version": "0.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "accepts": { 8 | "version": "1.3.7", 9 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 10 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 11 | "requires": { 12 | "mime-types": "~2.1.24", 13 | "negotiator": "0.6.2" 14 | } 15 | }, 16 | "acorn": { 17 | "version": "2.7.0", 18 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-2.7.0.tgz", 19 | "integrity": "sha1-q259nYhqrKiwhbwzEreaGYQz8Oc=" 20 | }, 21 | "acorn-globals": { 22 | "version": "1.0.9", 23 | "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-1.0.9.tgz", 24 | "integrity": "sha1-VbtemGkVB7dFedBRNBMhfDgMVM8=", 25 | "requires": { 26 | "acorn": "^2.1.0" 27 | } 28 | }, 29 | "align-text": { 30 | "version": "0.1.4", 31 | "resolved": "https://registry.npmjs.org/align-text/-/align-text-0.1.4.tgz", 32 | "integrity": "sha1-DNkKVhCT810KmSVsIrcGlDP60Rc=", 33 | "requires": { 34 | "kind-of": "^3.0.2", 35 | "longest": "^1.0.1", 36 | "repeat-string": "^1.5.2" 37 | } 38 | }, 39 | "amdefine": { 40 | "version": "1.0.1", 41 | "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", 42 | "integrity": "sha1-SlKCrBZHKek2Gbz9OtFR+BfOkfU=" 43 | }, 44 | "array-flatten": { 45 | "version": "1.1.1", 46 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 47 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 48 | }, 49 | "asap": { 50 | "version": "1.0.0", 51 | "resolved": "https://registry.npmjs.org/asap/-/asap-1.0.0.tgz", 52 | "integrity": "sha1-sqRdpf36ILBJb8N2jMJ8EvqRan0=" 53 | }, 54 | "basic-auth": { 55 | "version": "2.0.1", 56 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 57 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 58 | "requires": { 59 | "safe-buffer": "5.1.2" 60 | } 61 | }, 62 | "body-parser": { 63 | "version": "1.18.3", 64 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.3.tgz", 65 | "integrity": "sha1-WykhmP/dVTs6DyDe0FkrlWlVyLQ=", 66 | "requires": { 67 | "bytes": "3.0.0", 68 | "content-type": "~1.0.4", 69 | "debug": "2.6.9", 70 | "depd": "~1.1.2", 71 | "http-errors": "~1.6.3", 72 | "iconv-lite": "0.4.23", 73 | "on-finished": "~2.3.0", 74 | "qs": "6.5.2", 75 | "raw-body": "2.3.3", 76 | "type-is": "~1.6.16" 77 | } 78 | }, 79 | "bytes": { 80 | "version": "3.0.0", 81 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 82 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 83 | }, 84 | "camelcase": { 85 | "version": "1.2.1", 86 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-1.2.1.tgz", 87 | "integrity": "sha1-m7UwTS4LVmmLLHWLCKPqqdqlijk=" 88 | }, 89 | "center-align": { 90 | "version": "0.1.3", 91 | "resolved": "https://registry.npmjs.org/center-align/-/center-align-0.1.3.tgz", 92 | "integrity": "sha1-qg0yYptu6XIgBBHL1EYckHvCt60=", 93 | "requires": { 94 | "align-text": "^0.1.3", 95 | "lazy-cache": "^1.0.3" 96 | } 97 | }, 98 | "character-parser": { 99 | "version": "1.2.1", 100 | "resolved": "https://registry.npmjs.org/character-parser/-/character-parser-1.2.1.tgz", 101 | "integrity": "sha1-wN3kqxgnE7kZuXCVmhI+zBow/NY=" 102 | }, 103 | "clean-css": { 104 | "version": "3.4.28", 105 | "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-3.4.28.tgz", 106 | "integrity": "sha1-vxlF6C/ICPVWlebd6uwBQA79A/8=", 107 | "requires": { 108 | "commander": "2.8.x", 109 | "source-map": "0.4.x" 110 | }, 111 | "dependencies": { 112 | "commander": { 113 | "version": "2.8.1", 114 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.8.1.tgz", 115 | "integrity": "sha1-Br42f+v9oMMwqh4qBy09yXYkJdQ=", 116 | "requires": { 117 | "graceful-readlink": ">= 1.0.0" 118 | } 119 | } 120 | } 121 | }, 122 | "cliui": { 123 | "version": "2.1.0", 124 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-2.1.0.tgz", 125 | "integrity": "sha1-S0dXYP+AJkx2LDoXGQMukcf+oNE=", 126 | "requires": { 127 | "center-align": "^0.1.1", 128 | "right-align": "^0.1.1", 129 | "wordwrap": "0.0.2" 130 | }, 131 | "dependencies": { 132 | "wordwrap": { 133 | "version": "0.0.2", 134 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.2.tgz", 135 | "integrity": "sha1-t5Zpu0LstAn4PVg8rVLKF+qhZD8=" 136 | } 137 | } 138 | }, 139 | "commander": { 140 | "version": "2.6.0", 141 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.6.0.tgz", 142 | "integrity": "sha1-nfflL7Kgyw+4kFjugMMQQiXzfh0=" 143 | }, 144 | "constantinople": { 145 | "version": "3.0.2", 146 | "resolved": "https://registry.npmjs.org/constantinople/-/constantinople-3.0.2.tgz", 147 | "integrity": "sha1-S5RdmTeQe82Y7ldRIsOBdRZUQUE=", 148 | "requires": { 149 | "acorn": "^2.1.0" 150 | } 151 | }, 152 | "content-disposition": { 153 | "version": "0.5.2", 154 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 155 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 156 | }, 157 | "content-type": { 158 | "version": "1.0.4", 159 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 160 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 161 | }, 162 | "cookie": { 163 | "version": "0.3.1", 164 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 165 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 166 | }, 167 | "cookie-parser": { 168 | "version": "1.4.4", 169 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.4.tgz", 170 | "integrity": "sha512-lo13tqF3JEtFO7FyA49CqbhaFkskRJ0u/UAiINgrIXeRCY41c88/zxtrECl8AKH3B0hj9q10+h3Kt8I7KlW4tw==", 171 | "requires": { 172 | "cookie": "0.3.1", 173 | "cookie-signature": "1.0.6" 174 | } 175 | }, 176 | "cookie-signature": { 177 | "version": "1.0.6", 178 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 179 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 180 | }, 181 | "crypto": { 182 | "version": "1.0.1", 183 | "resolved": "https://registry.npmjs.org/crypto/-/crypto-1.0.1.tgz", 184 | "integrity": "sha512-VxBKmeNcqQdiUQUW2Tzq0t377b54N2bMtXO/qiLa+6eRRmmC4qT3D4OnTGoT/U6O9aklQ/jTwbOtRMTTY8G0Ig==" 185 | }, 186 | "css": { 187 | "version": "1.0.8", 188 | "resolved": "https://registry.npmjs.org/css/-/css-1.0.8.tgz", 189 | "integrity": "sha1-k4aBHKgrzMnuf7WnMrHioxfIo+c=", 190 | "requires": { 191 | "css-parse": "1.0.4", 192 | "css-stringify": "1.0.5" 193 | } 194 | }, 195 | "css-parse": { 196 | "version": "1.0.4", 197 | "resolved": "https://registry.npmjs.org/css-parse/-/css-parse-1.0.4.tgz", 198 | "integrity": "sha1-OLBQP7+dqfVOnB29pg4UXHcRe90=" 199 | }, 200 | "css-stringify": { 201 | "version": "1.0.5", 202 | "resolved": "https://registry.npmjs.org/css-stringify/-/css-stringify-1.0.5.tgz", 203 | "integrity": "sha1-sNBClG2ylTu50pKQCmy19tASIDE=" 204 | }, 205 | "debug": { 206 | "version": "2.6.9", 207 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 208 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 209 | "requires": { 210 | "ms": "2.0.0" 211 | } 212 | }, 213 | "decamelize": { 214 | "version": "1.2.0", 215 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 216 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" 217 | }, 218 | "depd": { 219 | "version": "1.1.2", 220 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 221 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 222 | }, 223 | "destroy": { 224 | "version": "1.0.4", 225 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 226 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 227 | }, 228 | "ee-first": { 229 | "version": "1.1.1", 230 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 231 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 232 | }, 233 | "encodeurl": { 234 | "version": "1.0.2", 235 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 236 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 237 | }, 238 | "escape-html": { 239 | "version": "1.0.3", 240 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 241 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 242 | }, 243 | "etag": { 244 | "version": "1.8.1", 245 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 246 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 247 | }, 248 | "express": { 249 | "version": "4.16.4", 250 | "resolved": "https://registry.npmjs.org/express/-/express-4.16.4.tgz", 251 | "integrity": "sha512-j12Uuyb4FMrd/qQAm6uCHAkPtO8FDTRJZBDd5D2KOL2eLaz1yUNdUB/NOIyq0iU4q4cFarsUCrnFDPBcnksuOg==", 252 | "requires": { 253 | "accepts": "~1.3.5", 254 | "array-flatten": "1.1.1", 255 | "body-parser": "1.18.3", 256 | "content-disposition": "0.5.2", 257 | "content-type": "~1.0.4", 258 | "cookie": "0.3.1", 259 | "cookie-signature": "1.0.6", 260 | "debug": "2.6.9", 261 | "depd": "~1.1.2", 262 | "encodeurl": "~1.0.2", 263 | "escape-html": "~1.0.3", 264 | "etag": "~1.8.1", 265 | "finalhandler": "1.1.1", 266 | "fresh": "0.5.2", 267 | "merge-descriptors": "1.0.1", 268 | "methods": "~1.1.2", 269 | "on-finished": "~2.3.0", 270 | "parseurl": "~1.3.2", 271 | "path-to-regexp": "0.1.7", 272 | "proxy-addr": "~2.0.4", 273 | "qs": "6.5.2", 274 | "range-parser": "~1.2.0", 275 | "safe-buffer": "5.1.2", 276 | "send": "0.16.2", 277 | "serve-static": "1.13.2", 278 | "setprototypeof": "1.1.0", 279 | "statuses": "~1.4.0", 280 | "type-is": "~1.6.16", 281 | "utils-merge": "1.0.1", 282 | "vary": "~1.1.2" 283 | } 284 | }, 285 | "finalhandler": { 286 | "version": "1.1.1", 287 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", 288 | "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", 289 | "requires": { 290 | "debug": "2.6.9", 291 | "encodeurl": "~1.0.2", 292 | "escape-html": "~1.0.3", 293 | "on-finished": "~2.3.0", 294 | "parseurl": "~1.3.2", 295 | "statuses": "~1.4.0", 296 | "unpipe": "~1.0.0" 297 | } 298 | }, 299 | "forwarded": { 300 | "version": "0.1.2", 301 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 302 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 303 | }, 304 | "fresh": { 305 | "version": "0.5.2", 306 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 307 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 308 | }, 309 | "fs-extra": { 310 | "version": "8.0.1", 311 | "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.0.1.tgz", 312 | "integrity": "sha512-W+XLrggcDzlle47X/XnS7FXrXu9sDo+Ze9zpndeBxdgv88FHLm1HtmkhEwavruS6koanBjp098rUpHs65EmG7A==", 313 | "requires": { 314 | "graceful-fs": "^4.1.2", 315 | "jsonfile": "^4.0.0", 316 | "universalify": "^0.1.0" 317 | } 318 | }, 319 | "graceful-fs": { 320 | "version": "4.1.15", 321 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.15.tgz", 322 | "integrity": "sha512-6uHUhOPEBgQ24HM+r6b/QwWfZq+yiFcipKFrOFiBEnWdy5sdzYoi+pJeQaPI5qOLRFqWmAXUPQNsielzdLoecA==" 323 | }, 324 | "graceful-readlink": { 325 | "version": "1.0.1", 326 | "resolved": "https://registry.npmjs.org/graceful-readlink/-/graceful-readlink-1.0.1.tgz", 327 | "integrity": "sha1-TK+tdrxi8C+gObL5Tpo906ORpyU=" 328 | }, 329 | "http-errors": { 330 | "version": "1.6.3", 331 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", 332 | "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", 333 | "requires": { 334 | "depd": "~1.1.2", 335 | "inherits": "2.0.3", 336 | "setprototypeof": "1.1.0", 337 | "statuses": ">= 1.4.0 < 2" 338 | } 339 | }, 340 | "iconv-lite": { 341 | "version": "0.4.23", 342 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", 343 | "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", 344 | "requires": { 345 | "safer-buffer": ">= 2.1.2 < 3" 346 | } 347 | }, 348 | "inherits": { 349 | "version": "2.0.3", 350 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 351 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 352 | }, 353 | "ipaddr.js": { 354 | "version": "1.9.0", 355 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz", 356 | "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA==" 357 | }, 358 | "is-buffer": { 359 | "version": "1.1.6", 360 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", 361 | "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" 362 | }, 363 | "is-promise": { 364 | "version": "2.1.0", 365 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", 366 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=" 367 | }, 368 | "jade": { 369 | "version": "1.11.0", 370 | "resolved": "https://registry.npmjs.org/jade/-/jade-1.11.0.tgz", 371 | "integrity": "sha1-nIDlOMEtP7lcjZu5VZ+gzAQEBf0=", 372 | "requires": { 373 | "character-parser": "1.2.1", 374 | "clean-css": "^3.1.9", 375 | "commander": "~2.6.0", 376 | "constantinople": "~3.0.1", 377 | "jstransformer": "0.0.2", 378 | "mkdirp": "~0.5.0", 379 | "transformers": "2.1.0", 380 | "uglify-js": "^2.4.19", 381 | "void-elements": "~2.0.1", 382 | "with": "~4.0.0" 383 | } 384 | }, 385 | "jsonfile": { 386 | "version": "4.0.0", 387 | "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", 388 | "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", 389 | "requires": { 390 | "graceful-fs": "^4.1.6" 391 | } 392 | }, 393 | "jstransformer": { 394 | "version": "0.0.2", 395 | "resolved": "https://registry.npmjs.org/jstransformer/-/jstransformer-0.0.2.tgz", 396 | "integrity": "sha1-eq4pqQPRls+glz2IXT5HlH7Ndqs=", 397 | "requires": { 398 | "is-promise": "^2.0.0", 399 | "promise": "^6.0.1" 400 | } 401 | }, 402 | "kind-of": { 403 | "version": "3.2.2", 404 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 405 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 406 | "requires": { 407 | "is-buffer": "^1.1.5" 408 | } 409 | }, 410 | "lazy-cache": { 411 | "version": "1.0.4", 412 | "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-1.0.4.tgz", 413 | "integrity": "sha1-odePw6UEdMuAhF07O24dpJpEbo4=" 414 | }, 415 | "longest": { 416 | "version": "1.0.1", 417 | "resolved": "https://registry.npmjs.org/longest/-/longest-1.0.1.tgz", 418 | "integrity": "sha1-MKCy2jj3N3DoKUoNIuZiXtd9AJc=" 419 | }, 420 | "media-typer": { 421 | "version": "0.3.0", 422 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 423 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 424 | }, 425 | "merge-descriptors": { 426 | "version": "1.0.1", 427 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 428 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 429 | }, 430 | "methods": { 431 | "version": "1.1.2", 432 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 433 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 434 | }, 435 | "mime": { 436 | "version": "1.4.1", 437 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz", 438 | "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==" 439 | }, 440 | "mime-db": { 441 | "version": "1.40.0", 442 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.40.0.tgz", 443 | "integrity": "sha512-jYdeOMPy9vnxEqFRRo6ZvTZ8d9oPb+k18PKoYNYUe2stVEBPPwsln/qWzdbmaIvnhZ9v2P+CuecK+fpUfsV2mA==" 444 | }, 445 | "mime-types": { 446 | "version": "2.1.24", 447 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.24.tgz", 448 | "integrity": "sha512-WaFHS3MCl5fapm3oLxU4eYDw77IQM2ACcxQ9RIxfaC3ooc6PFuBMGZZsYpvoXS5D5QTWPieo1jjLdAm3TBP3cQ==", 449 | "requires": { 450 | "mime-db": "1.40.0" 451 | } 452 | }, 453 | "minimist": { 454 | "version": "0.0.8", 455 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 456 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" 457 | }, 458 | "mkdirp": { 459 | "version": "0.5.1", 460 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 461 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 462 | "requires": { 463 | "minimist": "0.0.8" 464 | } 465 | }, 466 | "morgan": { 467 | "version": "1.9.1", 468 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.1.tgz", 469 | "integrity": "sha512-HQStPIV4y3afTiCYVxirakhlCfGkI161c76kKFca7Fk1JusM//Qeo1ej2XaMniiNeaZklMVrh3vTtIzpzwbpmA==", 470 | "requires": { 471 | "basic-auth": "~2.0.0", 472 | "debug": "2.6.9", 473 | "depd": "~1.1.2", 474 | "on-finished": "~2.3.0", 475 | "on-headers": "~1.0.1" 476 | } 477 | }, 478 | "ms": { 479 | "version": "2.0.0", 480 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 481 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 482 | }, 483 | "negotiator": { 484 | "version": "0.6.2", 485 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 486 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 487 | }, 488 | "on-finished": { 489 | "version": "2.3.0", 490 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 491 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 492 | "requires": { 493 | "ee-first": "1.1.1" 494 | } 495 | }, 496 | "on-headers": { 497 | "version": "1.0.2", 498 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 499 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 500 | }, 501 | "optimist": { 502 | "version": "0.3.7", 503 | "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.3.7.tgz", 504 | "integrity": "sha1-yQlBrVnkJzMokjB00s8ufLxuwNk=", 505 | "requires": { 506 | "wordwrap": "~0.0.2" 507 | } 508 | }, 509 | "parseurl": { 510 | "version": "1.3.3", 511 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 512 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 513 | }, 514 | "path-to-regexp": { 515 | "version": "0.1.7", 516 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 517 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 518 | }, 519 | "promise": { 520 | "version": "6.1.0", 521 | "resolved": "https://registry.npmjs.org/promise/-/promise-6.1.0.tgz", 522 | "integrity": "sha1-LOcp9rlLRcJoka0GAsXJDgTG7vY=", 523 | "requires": { 524 | "asap": "~1.0.0" 525 | } 526 | }, 527 | "proxy-addr": { 528 | "version": "2.0.5", 529 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz", 530 | "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==", 531 | "requires": { 532 | "forwarded": "~0.1.2", 533 | "ipaddr.js": "1.9.0" 534 | } 535 | }, 536 | "qs": { 537 | "version": "6.5.2", 538 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", 539 | "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" 540 | }, 541 | "range-parser": { 542 | "version": "1.2.1", 543 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 544 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 545 | }, 546 | "raw-body": { 547 | "version": "2.3.3", 548 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.3.tgz", 549 | "integrity": "sha512-9esiElv1BrZoI3rCDuOuKCBRbuApGGaDPQfjSflGxdy4oyzqghxu6klEkkVIvBje+FF0BX9coEv8KqW6X/7njw==", 550 | "requires": { 551 | "bytes": "3.0.0", 552 | "http-errors": "1.6.3", 553 | "iconv-lite": "0.4.23", 554 | "unpipe": "1.0.0" 555 | } 556 | }, 557 | "repeat-string": { 558 | "version": "1.6.1", 559 | "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", 560 | "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=" 561 | }, 562 | "right-align": { 563 | "version": "0.1.3", 564 | "resolved": "https://registry.npmjs.org/right-align/-/right-align-0.1.3.tgz", 565 | "integrity": "sha1-YTObci/mo1FWiSENJOFMlhSGE+8=", 566 | "requires": { 567 | "align-text": "^0.1.1" 568 | } 569 | }, 570 | "safe-buffer": { 571 | "version": "5.1.2", 572 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 573 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 574 | }, 575 | "safer-buffer": { 576 | "version": "2.1.2", 577 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 578 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 579 | }, 580 | "send": { 581 | "version": "0.16.2", 582 | "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", 583 | "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", 584 | "requires": { 585 | "debug": "2.6.9", 586 | "depd": "~1.1.2", 587 | "destroy": "~1.0.4", 588 | "encodeurl": "~1.0.2", 589 | "escape-html": "~1.0.3", 590 | "etag": "~1.8.1", 591 | "fresh": "0.5.2", 592 | "http-errors": "~1.6.2", 593 | "mime": "1.4.1", 594 | "ms": "2.0.0", 595 | "on-finished": "~2.3.0", 596 | "range-parser": "~1.2.0", 597 | "statuses": "~1.4.0" 598 | } 599 | }, 600 | "serve-static": { 601 | "version": "1.13.2", 602 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz", 603 | "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==", 604 | "requires": { 605 | "encodeurl": "~1.0.2", 606 | "escape-html": "~1.0.3", 607 | "parseurl": "~1.3.2", 608 | "send": "0.16.2" 609 | } 610 | }, 611 | "setprototypeof": { 612 | "version": "1.1.0", 613 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", 614 | "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==" 615 | }, 616 | "source-map": { 617 | "version": "0.4.4", 618 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.4.4.tgz", 619 | "integrity": "sha1-66T12pwNyZneaAMti092FzZSA2s=", 620 | "requires": { 621 | "amdefine": ">=0.0.4" 622 | } 623 | }, 624 | "statuses": { 625 | "version": "1.4.0", 626 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", 627 | "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==" 628 | }, 629 | "transformers": { 630 | "version": "2.1.0", 631 | "resolved": "https://registry.npmjs.org/transformers/-/transformers-2.1.0.tgz", 632 | "integrity": "sha1-XSPLNVYd2F3Gf7hIIwm0fVPM6ac=", 633 | "requires": { 634 | "css": "~1.0.8", 635 | "promise": "~2.0", 636 | "uglify-js": "~2.2.5" 637 | }, 638 | "dependencies": { 639 | "is-promise": { 640 | "version": "1.0.1", 641 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-1.0.1.tgz", 642 | "integrity": "sha1-MVc3YcBX4zwukaq56W2gjO++duU=" 643 | }, 644 | "promise": { 645 | "version": "2.0.0", 646 | "resolved": "https://registry.npmjs.org/promise/-/promise-2.0.0.tgz", 647 | "integrity": "sha1-RmSKqdYFr10ucMMCS/WUNtoCuA4=", 648 | "requires": { 649 | "is-promise": "~1" 650 | } 651 | }, 652 | "source-map": { 653 | "version": "0.1.43", 654 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.1.43.tgz", 655 | "integrity": "sha1-wkvBRspRfBRx9drL4lcbK3+eM0Y=", 656 | "requires": { 657 | "amdefine": ">=0.0.4" 658 | } 659 | }, 660 | "uglify-js": { 661 | "version": "2.2.5", 662 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.2.5.tgz", 663 | "integrity": "sha1-puAqcNg5eSuXgEiLe4sYTAlcmcc=", 664 | "requires": { 665 | "optimist": "~0.3.5", 666 | "source-map": "~0.1.7" 667 | } 668 | } 669 | } 670 | }, 671 | "type-is": { 672 | "version": "1.6.18", 673 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 674 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 675 | "requires": { 676 | "media-typer": "0.3.0", 677 | "mime-types": "~2.1.24" 678 | } 679 | }, 680 | "uglify-js": { 681 | "version": "2.8.29", 682 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.8.29.tgz", 683 | "integrity": "sha1-KcVzMUgFe7Th913zW3qcty5qWd0=", 684 | "requires": { 685 | "source-map": "~0.5.1", 686 | "uglify-to-browserify": "~1.0.0", 687 | "yargs": "~3.10.0" 688 | }, 689 | "dependencies": { 690 | "source-map": { 691 | "version": "0.5.7", 692 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", 693 | "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" 694 | } 695 | } 696 | }, 697 | "uglify-to-browserify": { 698 | "version": "1.0.2", 699 | "resolved": "https://registry.npmjs.org/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz", 700 | "integrity": "sha1-bgkk1r2mta/jSeOabWMoUKD4grc=", 701 | "optional": true 702 | }, 703 | "universalify": { 704 | "version": "0.1.2", 705 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", 706 | "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==" 707 | }, 708 | "unpipe": { 709 | "version": "1.0.0", 710 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 711 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 712 | }, 713 | "utils-merge": { 714 | "version": "1.0.1", 715 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 716 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 717 | }, 718 | "vary": { 719 | "version": "1.1.2", 720 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 721 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 722 | }, 723 | "void-elements": { 724 | "version": "2.0.1", 725 | "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz", 726 | "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w=" 727 | }, 728 | "window-size": { 729 | "version": "0.1.0", 730 | "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.1.0.tgz", 731 | "integrity": "sha1-VDjNLqk7IC76Ohn+iIeu58lPnJ0=" 732 | }, 733 | "with": { 734 | "version": "4.0.3", 735 | "resolved": "https://registry.npmjs.org/with/-/with-4.0.3.tgz", 736 | "integrity": "sha1-7v0VTp550sjTQXtkeo8U2f7M4U4=", 737 | "requires": { 738 | "acorn": "^1.0.1", 739 | "acorn-globals": "^1.0.3" 740 | }, 741 | "dependencies": { 742 | "acorn": { 743 | "version": "1.2.2", 744 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-1.2.2.tgz", 745 | "integrity": "sha1-yM4n3grMdtiW0rH6099YjZ6C8BQ=" 746 | } 747 | } 748 | }, 749 | "wordwrap": { 750 | "version": "0.0.3", 751 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", 752 | "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=" 753 | }, 754 | "yargs": { 755 | "version": "3.10.0", 756 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-3.10.0.tgz", 757 | "integrity": "sha1-9+572FfdfB0tOMDnTvvWgdFDH9E=", 758 | "requires": { 759 | "camelcase": "^1.0.2", 760 | "cliui": "^2.1.0", 761 | "decamelize": "^1.0.0", 762 | "window-size": "0.1.0" 763 | } 764 | } 765 | } 766 | } 767 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "express-example", 3 | "version": "0.0.0", 4 | "private": true, 5 | "scripts": { 6 | "start": "node ./bin/www" 7 | }, 8 | "dependencies": { 9 | "cookie-parser": "~1.4.4", 10 | "crypto": "^1.0.1", 11 | "debug": "~2.6.9", 12 | "express": "~4.16.1", 13 | "fs-extra": "^8.0.1", 14 | "http-errors": "~1.6.3", 15 | "jade": "~1.11.0", 16 | "morgan": "~1.9.1" 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /public/stylesheets/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 50px; 3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif; 4 | } 5 | 6 | a { 7 | color: #00B7FF; 8 | } 9 | -------------------------------------------------------------------------------- /routes/index.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | var exec = require('child_process').exec; 4 | var fs = require("fs-extra"); 5 | var crypto = require('crypto'); 6 | 7 | router.post('/run/:language', function(req, res, next) { 8 | var content = req.body.content; 9 | 10 | var folder = "/tmp/" + crypto.randomBytes(16).toString('hex'); 11 | fs.mkdirSync(`${folder}`); 12 | 13 | if (req.params.language == "php") { 14 | var inputFile = `${folder}/main.php`; 15 | fs.writeFileSync(inputFile, content); 16 | var command = `docker run --rm -v ${folder}:${folder} ntcd_php php ${inputFile} ${folder}/output`; 17 | } else if (req.params.language == "javascript") { 18 | var inputFile = `${folder}/main.js`; 19 | fs.writeFileSync(inputFile, content); 20 | var command = `docker run --rm -v ${folder}:${folder} ntcd_javascript node ${inputFile} ${folder}/output`; 21 | } 22 | 23 | if (command == null) { 24 | res.json({ 25 | stdout: "", 26 | error: "language not support", 27 | }); 28 | } 29 | 30 | exec(command, 31 | (error, exec_time, stderr) => { 32 | var stdout = fs.readFileSync(`${folder}/output`, 'utf8'); 33 | res.json({ 34 | stdout: stdout, 35 | error: stderr, 36 | time: exec_time 37 | }); 38 | 39 | fs.remove(folder, (err) => {}); 40 | }); 41 | }); 42 | 43 | module.exports = router; 44 | -------------------------------------------------------------------------------- /views/error.jade: -------------------------------------------------------------------------------- 1 | extends layout 2 | 3 | block content 4 | h1= message 5 | h2= error.status 6 | pre #{error.stack} 7 | -------------------------------------------------------------------------------- /views/index.jade: -------------------------------------------------------------------------------- 1 | extends layout 2 | 3 | block content 4 | h1= title 5 | p Welcome to #{title} 6 | -------------------------------------------------------------------------------- /views/layout.jade: -------------------------------------------------------------------------------- 1 | doctype html 2 | html 3 | head 4 | title= title 5 | link(rel='stylesheet', href='/stylesheets/style.css') 6 | body 7 | block content 8 | --------------------------------------------------------------------------------