mirror of
				https://github.com/engineer-man/piston.git
				synced 2025-10-22 18:40:03 +02:00 
			
		
		
		
	Compare commits
	
		
			No commits in common. "286fb5741551c54705a3170ba40a7411107c18e3" and "d1315b11962b1f4867c640eafb25e14df9e4622b" have entirely different histories.
		
	
	
		
			286fb57415
			...
			d1315b1196
		
	
		
					 11 changed files with 403 additions and 437 deletions
				
			
		
							
								
								
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -2,4 +2,3 @@ data/ | ||||||
| .piston_env | .piston_env | ||||||
| node_modules | node_modules | ||||||
| result | result | ||||||
| .vscode/ |  | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ const events = require('events'); | ||||||
| 
 | 
 | ||||||
| const runtime = require('../runtime'); | const runtime = require('../runtime'); | ||||||
| const { Job } = require('../job'); | const { Job } = require('../job'); | ||||||
|  | const logger = require('logplease').create('api/v3'); | ||||||
| 
 | 
 | ||||||
| const SIGNALS = [ | const SIGNALS = [ | ||||||
|     'SIGABRT', |     'SIGABRT', | ||||||
|  | @ -80,9 +81,49 @@ function get_job(body) { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         const rt = runtime.find(rt => |         if (compile_memory_limit) { | ||||||
|             [...rt.aliases, rt.language].includes(rt.language) |             if (typeof compile_memory_limit !== 'number') { | ||||||
|         ); |                 return reject({ | ||||||
|  |                     message: 'if specified, compile_memory_limit must be a number', | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if ( | ||||||
|  |                 config.compile_memory_limit >= 0 && | ||||||
|  |                 (compile_memory_limit > config.compile_memory_limit || | ||||||
|  |                     compile_memory_limit < 0) | ||||||
|  |             ) { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: | ||||||
|  |                         'compile_memory_limit cannot exceed the configured limit of ' + | ||||||
|  |                         config.compile_memory_limit, | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (run_memory_limit) { | ||||||
|  |             if (typeof run_memory_limit !== 'number') { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: 'if specified, run_memory_limit must be a number', | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if ( | ||||||
|  |                 config.run_memory_limit >= 0 && | ||||||
|  |                 (run_memory_limit > config.run_memory_limit || run_memory_limit < 0) | ||||||
|  |             ) { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: | ||||||
|  |                         'run_memory_limit cannot exceed the configured limit of ' + | ||||||
|  |                         config.run_memory_limit, | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const rt = runtime.find(rt => [ | ||||||
|  |             ...rt.aliases, | ||||||
|  |             rt.language | ||||||
|  |         ].includes(rt.language)) | ||||||
| 
 | 
 | ||||||
|         if (rt === undefined) { |         if (rt === undefined) { | ||||||
|             return reject({ |             return reject({ | ||||||
|  |  | ||||||
|  | @ -3,49 +3,12 @@ const router = express.Router(); | ||||||
| 
 | 
 | ||||||
| const events = require('events'); | const events = require('events'); | ||||||
| 
 | 
 | ||||||
|  | const config = require('../config'); | ||||||
| const runtime = require('../runtime'); | const runtime = require('../runtime'); | ||||||
| const { Job } = require('../job'); | const { Job } = require('../job'); | ||||||
|  | const logger = require('logplease').create('api/v3'); | ||||||
| 
 | 
 | ||||||
| const SIGNALS = [ | const SIGNALS = ["SIGABRT","SIGALRM","SIGBUS","SIGCHLD","SIGCLD","SIGCONT","SIGEMT","SIGFPE","SIGHUP","SIGILL","SIGINFO","SIGINT","SIGIO","SIGIOT","SIGKILL","SIGLOST","SIGPIPE","SIGPOLL","SIGPROF","SIGPWR","SIGQUIT","SIGSEGV","SIGSTKFLT","SIGSTOP","SIGTSTP","SIGSYS","SIGTERM","SIGTRAP","SIGTTIN","SIGTTOU","SIGUNUSED","SIGURG","SIGUSR1","SIGUSR2","SIGVTALRM","SIGXCPU","SIGXFSZ","SIGWINCH"] | ||||||
|     'SIGABRT', |  | ||||||
|     'SIGALRM', |  | ||||||
|     'SIGBUS', |  | ||||||
|     'SIGCHLD', |  | ||||||
|     'SIGCLD', |  | ||||||
|     'SIGCONT', |  | ||||||
|     'SIGEMT', |  | ||||||
|     'SIGFPE', |  | ||||||
|     'SIGHUP', |  | ||||||
|     'SIGILL', |  | ||||||
|     'SIGINFO', |  | ||||||
|     'SIGINT', |  | ||||||
|     'SIGIO', |  | ||||||
|     'SIGIOT', |  | ||||||
|     'SIGKILL', |  | ||||||
|     'SIGLOST', |  | ||||||
|     'SIGPIPE', |  | ||||||
|     'SIGPOLL', |  | ||||||
|     'SIGPROF', |  | ||||||
|     'SIGPWR', |  | ||||||
|     'SIGQUIT', |  | ||||||
|     'SIGSEGV', |  | ||||||
|     'SIGSTKFLT', |  | ||||||
|     'SIGSTOP', |  | ||||||
|     'SIGTSTP', |  | ||||||
|     'SIGSYS', |  | ||||||
|     'SIGTERM', |  | ||||||
|     'SIGTRAP', |  | ||||||
|     'SIGTTIN', |  | ||||||
|     'SIGTTOU', |  | ||||||
|     'SIGUNUSED', |  | ||||||
|     'SIGURG', |  | ||||||
|     'SIGUSR1', |  | ||||||
|     'SIGUSR2', |  | ||||||
|     'SIGVTALRM', |  | ||||||
|     'SIGXCPU', |  | ||||||
|     'SIGXFSZ', |  | ||||||
|     'SIGWINCH', |  | ||||||
| ]; |  | ||||||
| // ref: https://man7.org/linux/man-pages/man7/signal.7.html
 | // ref: https://man7.org/linux/man-pages/man7/signal.7.html
 | ||||||
| 
 | 
 | ||||||
| function get_job(body){ | function get_job(body){ | ||||||
|  | @ -57,96 +20,93 @@ function get_job(body) { | ||||||
|         compile_memory_limit, |         compile_memory_limit, | ||||||
|         run_memory_limit, |         run_memory_limit, | ||||||
|         run_timeout, |         run_timeout, | ||||||
|         compile_timeout, |         compile_timeout | ||||||
|     } = body; |     } = body; | ||||||
| 
 | 
 | ||||||
|     return new Promise((resolve, reject) => { |     return new Promise((resolve, reject) => { | ||||||
|         if (typeof runtime_id !== 'number') { |         if (typeof runtime_id !== 'number') { | ||||||
|             return reject({ |             return reject({ | ||||||
|                 message: 'runtime_id is required as a number', |                 message: 'runtime_id is required as a number' | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (!files || !Array.isArray(files)) { |         if (!Array.isArray(files)) { | ||||||
|             return reject({ |             return reject({ | ||||||
|                 message: 'files is required as an array', |                 message: 'files is required as an array', | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         for (const [i, file] of files.entries()) { | ||||||
|  |             if (typeof file.content !== 'string') { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: `files[${i}].content is required as a string`, | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (compile_memory_limit) { | ||||||
|  |             if (typeof compile_memory_limit !== 'number') { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: 'if specified, compile_memory_limit must be a number', | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if ( | ||||||
|  |                 config.compile_memory_limit >= 0 && | ||||||
|  |                 (compile_memory_limit > config.compile_memory_limit || | ||||||
|  |                     compile_memory_limit < 0) | ||||||
|  |             ) { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: | ||||||
|  |                         'compile_memory_limit cannot exceed the configured limit of ' + | ||||||
|  |                         config.compile_memory_limit, | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (run_memory_limit) { | ||||||
|  |             if (typeof run_memory_limit !== 'number') { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: 'if specified, run_memory_limit must be a number', | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if ( | ||||||
|  |                 config.run_memory_limit >= 0 && | ||||||
|  |                 (run_memory_limit > config.run_memory_limit || run_memory_limit < 0) | ||||||
|  |             ) { | ||||||
|  |                 return reject({ | ||||||
|  |                     message: | ||||||
|  |                         'run_memory_limit cannot exceed the configured limit of ' + | ||||||
|  |                         config.run_memory_limit, | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         const rt = runtime[runtime_id]; |         const rt = runtime[runtime_id]; | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|         if (rt === undefined) { |         if (rt === undefined) { | ||||||
|             return reject({ |             return reject({ | ||||||
|                 message: `Runtime #${runtime_id} is unknown`, |                 message: `Runtime #${runtime_id} is unknown`, | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if ( |         resolve(new Job({ | ||||||
|             rt.language !== 'file' && |  | ||||||
|             !files.some(file => !file.encoding || file.encoding === 'utf8') |  | ||||||
|         ) { |  | ||||||
|             return reject({ |  | ||||||
|                 message: 'files must include at least one utf8 encoded file', |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (files.some(file => typeof file.content !== 'string')) { |  | ||||||
|             return reject({ |  | ||||||
|                 message: 'file.content is required as a string', |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         for (const constraint of ['memory_limit', 'timeout']) { |  | ||||||
|             for (const type of ['compile', 'run']) { |  | ||||||
|                 const constraint_name = `${type}_${constraint}`; |  | ||||||
|                 const constraint_value = body[constraint_name]; |  | ||||||
|                 const configured_limit = rt[`${constraint}s`][type]; |  | ||||||
|                 if (!constraint_value) { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (typeof constraint_value !== 'number') { |  | ||||||
|                     return reject({ |  | ||||||
|                         message: `If specified, ${constraint_name} must be a number`, |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
|                 if (configured_limit <= 0) { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (constraint_value > configured_limit) { |  | ||||||
|                     return reject({ |  | ||||||
|                         message: `${constraint_name} cannot exceed the configured limit of ${configured_limit}`, |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
|                 if (constraint_value < 0) { |  | ||||||
|                     return reject({ |  | ||||||
|                         message: `${constraint_name} must be non-negative`, |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         const job_compile_timeout = compile_timeout || rt.timeouts.compile; |  | ||||||
|         const job_run_timeout = run_timeout || rt.timeouts.run; |  | ||||||
|         const job_compile_memory_limit = |  | ||||||
|             compile_memory_limit || rt.memory_limits.compile; |  | ||||||
|         const job_run_memory_limit = run_memory_limit || rt.memory_limits.run; |  | ||||||
|         resolve( |  | ||||||
|             new Job({ |  | ||||||
|             runtime: rt, |             runtime: rt, | ||||||
|             args: args || [], |             args: args || [], | ||||||
|                 stdin: stdin || '', |             stdin: stdin || "", | ||||||
|             files, |             files, | ||||||
|             timeouts: { |             timeouts: { | ||||||
|                     run: job_run_timeout, |                 run: run_timeout || 3000, | ||||||
|                     compile: job_compile_timeout, |                 compile: compile_timeout || 10000, | ||||||
|             }, |             }, | ||||||
|             memory_limits: { |             memory_limits: { | ||||||
|                     run: job_run_memory_limit, |                 run: run_memory_limit || config.run_memory_limit, | ||||||
|                     compile: job_compile_memory_limit, |                 compile: compile_memory_limit || config.compile_memory_limit, | ||||||
|                 }, |             } | ||||||
|  |         })); | ||||||
|     }) |     }) | ||||||
|         ); | 
 | ||||||
|     }); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| router.use((req, res, next) => { | router.use((req, res, next) => { | ||||||
|  | @ -164,106 +124,89 @@ router.use((req, res, next) => { | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| router.ws('/connect', async (ws, req) => { | router.ws('/connect', async (ws, req) => { | ||||||
|  | 
 | ||||||
|     let job = null; |     let job = null; | ||||||
|     let eventBus = new events.EventEmitter(); |     let eventBus = new events.EventEmitter(); | ||||||
| 
 | 
 | ||||||
|     eventBus.on('stdout', data => |     eventBus.on("stdout", (data) => ws.send(JSON.stringify({type: "data", stream: "stdout", data: data.toString()}))) | ||||||
|         ws.send( |     eventBus.on("stderr", (data) => ws.send(JSON.stringify({type: "data", stream: "stderr", data: data.toString()}))) | ||||||
|             JSON.stringify({ |     eventBus.on("stage", (stage)=> ws.send(JSON.stringify({type: "stage", stage}))) | ||||||
|                 type: 'data', |     eventBus.on("exit", (stage, status) => ws.send(JSON.stringify({type: "exit", stage, ...status}))) | ||||||
|                 stream: 'stdout', | 
 | ||||||
|                 data: data.toString(), |     ws.on("message", async (data) => { | ||||||
|             }) |  | ||||||
|         ) |  | ||||||
|     ); |  | ||||||
|     eventBus.on('stderr', data => |  | ||||||
|         ws.send( |  | ||||||
|             JSON.stringify({ |  | ||||||
|                 type: 'data', |  | ||||||
|                 stream: 'stderr', |  | ||||||
|                 data: data.toString(), |  | ||||||
|             }) |  | ||||||
|         ) |  | ||||||
|     ); |  | ||||||
|     eventBus.on('stage', stage => |  | ||||||
|         ws.send(JSON.stringify({ type: 'stage', stage })) |  | ||||||
|     ); |  | ||||||
|     eventBus.on('exit', (stage, status) => |  | ||||||
|         ws.send(JSON.stringify({ type: 'exit', stage, ...status })) |  | ||||||
|     ); |  | ||||||
| 
 | 
 | ||||||
|     ws.on('message', async data => { |  | ||||||
|         try{ |         try{ | ||||||
|             const msg = JSON.parse(data); |             const msg = JSON.parse(data); | ||||||
| 
 | 
 | ||||||
|             switch(msg.type){ |             switch(msg.type){ | ||||||
|                 case 'init': |                 case "init": | ||||||
|                     if(job === null){ |                     if(job === null){ | ||||||
|                         job = await get_job(msg); |                         job = await get_job(msg); | ||||||
| 
 | 
 | ||||||
|                         await job.prime(); |                         await job.prime(); | ||||||
| 
 | 
 | ||||||
|                         ws.send( |                         ws.send(JSON.stringify({ | ||||||
|                             JSON.stringify({ |                             type: "runtime", | ||||||
|                                 type: 'runtime', |  | ||||||
|                             language: job.runtime.language, |                             language: job.runtime.language, | ||||||
|                                 version: job.runtime.version.raw, |                             version: job.runtime.version.raw | ||||||
|                             }) |                         })) | ||||||
|                         ); |  | ||||||
| 
 | 
 | ||||||
|                         await job.execute_interactive(eventBus); |                         await job.execute_interactive(eventBus); | ||||||
| 
 | 
 | ||||||
|                         ws.close(4999, 'Job Completed'); |                         ws.close(4999, "Job Completed"); | ||||||
|  | 
 | ||||||
|                     }else{ |                     }else{ | ||||||
|                         ws.close(4000, 'Already Initialized'); |                         ws.close(4000, "Already Initialized"); | ||||||
|                     } |                     } | ||||||
|                     break; |                     break; | ||||||
|                 case 'data': |             case "data": | ||||||
|                 if(job !== null){ |                 if(job !== null){ | ||||||
|                         if (msg.stream === 'stdin') { |                     if(msg.stream === "stdin"){ | ||||||
|                             eventBus.emit('stdin', msg.data); |                         eventBus.emit("stdin", msg.data) | ||||||
|                     }else{ |                     }else{ | ||||||
|                             ws.close(4004, 'Can only write to stdin'); |                         ws.close(4004, "Can only write to stdin") | ||||||
|                     } |                     } | ||||||
|                 }else{ |                 }else{ | ||||||
|                         ws.close(4003, 'Not yet initialized'); |                     ws.close(4003, "Not yet initialized") | ||||||
|                 } |                 } | ||||||
|                 break; |                 break; | ||||||
|                 case 'signal': |             case "signal": | ||||||
|                 if(job !== null){ |                 if(job !== null){ | ||||||
|                     if(SIGNALS.includes(msg.signal)){ |                     if(SIGNALS.includes(msg.signal)){ | ||||||
|                             eventBus.emit('signal', msg.signal); |                         eventBus.emit("signal", msg.signal) | ||||||
|                     }else{ |                     }else{ | ||||||
|                             ws.close(4005, 'Invalid signal'); |                         ws.close(4005, "Invalid signal") | ||||||
|                     } |                     } | ||||||
|                 }else{ |                 }else{ | ||||||
|                         ws.close(4003, 'Not yet initialized'); |                     ws.close(4003, "Not yet initialized") | ||||||
|                 } |                 } | ||||||
|                 break; |                 break; | ||||||
|             } |             } | ||||||
|  | 
 | ||||||
|         }catch(error){ |         }catch(error){ | ||||||
|             ws.send(JSON.stringify({ type: 'error', message: error.message })); |             ws.send(JSON.stringify({type: "error", message: error.message})) | ||||||
|             ws.close(4002, 'Notified Error'); |             ws.close(4002, "Notified Error") | ||||||
|             // ws.close message is limited to 123 characters, so we notify over WS then close.
 |             // ws.close message is limited to 123 characters, so we notify over WS then close.
 | ||||||
|         } |         } | ||||||
|     }); |     }) | ||||||
| 
 | 
 | ||||||
|     ws.on('close', async () => { |     ws.on("close", async ()=>{ | ||||||
|         if(job !== null){ |         if(job !== null){ | ||||||
|             await job.cleanup(); |             await job.cleanup() | ||||||
|         } |         } | ||||||
|     }); |     }) | ||||||
| 
 | 
 | ||||||
|     setTimeout(()=>{ |     setTimeout(()=>{ | ||||||
|         //Terminate the socket after 1 second, if not initialized.
 |         //Terminate the socket after 1 second, if not initialized.
 | ||||||
|         if (job === null) ws.close(4001, 'Initialization Timeout'); |         if(job === null) | ||||||
|     }, 1000); |             ws.close(4001, "Initialization Timeout"); | ||||||
| }); |     }, 1000) | ||||||
|  | }) | ||||||
| 
 | 
 | ||||||
| router.post('/execute', async (req, res) => { | router.post('/execute', async (req, res) => { | ||||||
|  | 
 | ||||||
|     try{ |     try{ | ||||||
|         const job = await get_job(req.body); |         const job = await get_job(req.body); | ||||||
| 
 |  | ||||||
|         await job.prime(); |         await job.prime(); | ||||||
| 
 | 
 | ||||||
|         const result = await job.execute(); |         const result = await job.execute(); | ||||||
|  | @ -283,7 +226,7 @@ router.get('/runtimes', (req, res) => { | ||||||
|             version: rt.version.raw, |             version: rt.version.raw, | ||||||
|             aliases: rt.aliases, |             aliases: rt.aliases, | ||||||
|             runtime: rt.runtime, |             runtime: rt.runtime, | ||||||
|             id: rt.id, |             id: rt.id | ||||||
|         }; |         }; | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,149 +2,7 @@ const fss = require('fs'); | ||||||
| const Logger = require('logplease'); | const Logger = require('logplease'); | ||||||
| const logger = Logger.create('config'); | const logger = Logger.create('config'); | ||||||
| 
 | 
 | ||||||
| const options = { | function parse_overrides(overrides) { | ||||||
|     log_level: { |  | ||||||
|         desc: 'Level of data to log', |  | ||||||
|         default: 'INFO', |  | ||||||
|         validators: [ |  | ||||||
|             x => |  | ||||||
|                 Object.values(Logger.LogLevels).includes(x) || |  | ||||||
|                 `Log level ${x} does not exist`, |  | ||||||
|         ], |  | ||||||
|     }, |  | ||||||
|     bind_address: { |  | ||||||
|         desc: 'Address to bind REST API on', |  | ||||||
|         default: `0.0.0.0:${process.env['PORT'] || 2000}`, |  | ||||||
|         validators: [], |  | ||||||
|     }, |  | ||||||
|     data_directory: { |  | ||||||
|         desc: 'Absolute path to store all piston related data at', |  | ||||||
|         default: '/piston', |  | ||||||
|         validators: [ |  | ||||||
|             x => fss.exists_sync(x) || `Directory ${x} does not exist`, |  | ||||||
|         ], |  | ||||||
|     }, |  | ||||||
|     runner_uid_min: { |  | ||||||
|         desc: 'Minimum uid to use for runner', |  | ||||||
|         default: 1001, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     runner_uid_max: { |  | ||||||
|         desc: 'Maximum uid to use for runner', |  | ||||||
|         default: 1500, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     runner_gid_min: { |  | ||||||
|         desc: 'Minimum gid to use for runner', |  | ||||||
|         default: 1001, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     runner_gid_max: { |  | ||||||
|         desc: 'Maximum gid to use for runner', |  | ||||||
|         default: 1500, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     disable_networking: { |  | ||||||
|         desc: 'Set to true to disable networking', |  | ||||||
|         default: true, |  | ||||||
|         parser: x => x === 'true', |  | ||||||
|         validators: [x => typeof x === 'boolean' || `${x} is not a boolean`], |  | ||||||
|     }, |  | ||||||
|     output_max_size: { |  | ||||||
|         desc: 'Max size of each stdio buffer', |  | ||||||
|         default: 1024, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     max_process_count: { |  | ||||||
|         desc: 'Max number of processes per job', |  | ||||||
|         default: 64, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     max_open_files: { |  | ||||||
|         desc: 'Max number of open files per job', |  | ||||||
|         default: 2048, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     max_file_size: { |  | ||||||
|         desc: 'Max file size in bytes for a file', |  | ||||||
|         default: 10000000, //10MB
 |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     compile_timeout: { |  | ||||||
|         desc: 'Max time allowed for compile stage in milliseconds', |  | ||||||
|         default: 10000, // 10 seconds
 |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     run_timeout: { |  | ||||||
|         desc: 'Max time allowed for run stage in milliseconds', |  | ||||||
|         default: 3000, // 3 seconds
 |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     compile_memory_limit: { |  | ||||||
|         desc: 'Max memory usage for compile stage in bytes (set to -1 for no limit)', |  | ||||||
|         default: -1, // no limit
 |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     run_memory_limit: { |  | ||||||
|         desc: 'Max memory usage for run stage in bytes (set to -1 for no limit)', |  | ||||||
|         default: -1, // no limit
 |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], |  | ||||||
|     }, |  | ||||||
|     flake_path: { |  | ||||||
|         desc: 'Path to nix flake defining runtimes to install', |  | ||||||
|         default: 'github:engineer-man/piston?directory=packages', |  | ||||||
|         validators: [], |  | ||||||
|     }, |  | ||||||
|     runtime_set: { |  | ||||||
|         desc: 'Key on the flake specified by flake_path to access runtimes from', |  | ||||||
|         default: 'all', |  | ||||||
|         validators: [], |  | ||||||
|     }, |  | ||||||
|     max_concurrent_jobs: { |  | ||||||
|         desc: 'Maximum number of concurrent jobs to run at one time', |  | ||||||
|         default: 64, |  | ||||||
|         parser: parse_int, |  | ||||||
|         validators: [x => x > 0 || `${x} cannot be negative`], |  | ||||||
|     }, |  | ||||||
|     limit_overrides: { |  | ||||||
|         desc: 'Per-language exceptions in JSON format for each of:\ |  | ||||||
|         max_process_count, max_open_files, max_file_size, compile_memory_limit,\ |  | ||||||
|         run_memory_limit, compile_timeout, run_timeout, output_max_size', |  | ||||||
|         default: {}, |  | ||||||
|         parser: parse_overrides, |  | ||||||
|         validators: [ |  | ||||||
|             x => !!x || `Failed to parse the overrides\n${x}`, |  | ||||||
|             validate_overrides, |  | ||||||
|         ], |  | ||||||
|     }, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| Object.freeze(options); |  | ||||||
| 
 |  | ||||||
| function apply_validators(validators, validator_parameters) { |  | ||||||
|     for (const validator of validators) { |  | ||||||
|         const validation_response = validator(...validator_parameters); |  | ||||||
|         if (validation_response !== true) { |  | ||||||
|             return validation_response; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     return true; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function parse_overrides(overrides_string) { |  | ||||||
|     function get_parsed_json_or_null(overrides) { |  | ||||||
|     try { |     try { | ||||||
|         return JSON.parse(overrides); |         return JSON.parse(overrides); | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
|  | @ -152,13 +10,8 @@ function parse_overrides(overrides_string) { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|     const overrides = get_parsed_json_or_null(overrides_string); | function validate_overrides(overrides, options) { | ||||||
|     if (overrides === null) { |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
|     const parsed_overrides = {}; |  | ||||||
|     for (const language in overrides) { |     for (const language in overrides) { | ||||||
|         parsed_overrides[language] = {}; |  | ||||||
|         for (const key in overrides[language]) { |         for (const key in overrides[language]) { | ||||||
|             if ( |             if ( | ||||||
|                 ![ |                 ![ | ||||||
|  | @ -172,62 +25,223 @@ function parse_overrides(overrides_string) { | ||||||
|                     'output_max_size', |                     'output_max_size', | ||||||
|                 ].includes(key) |                 ].includes(key) | ||||||
|             ) { |             ) { | ||||||
|                 return null; |                 logger.error(`Invalid overridden option: ${key}`); | ||||||
|  |                 return false; | ||||||
|             } |             } | ||||||
|             // Find the option for the override
 |             const option = options.find(o => o.key === key); | ||||||
|             const option = options[key]; |  | ||||||
|             const parser = option.parser; |             const parser = option.parser; | ||||||
|             const raw_value = overrides[language][key]; |             const raw = overrides[language][key]; | ||||||
|             const parsed_value = parser(raw_value); |             const value = parser(raw); | ||||||
|             parsed_overrides[language][key] = parsed_value; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     return parsed_overrides; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function validate_overrides(overrides) { |  | ||||||
|     for (const language in overrides) { |  | ||||||
|         for (const key in overrides[language]) { |  | ||||||
|             const value = overrides[language][key]; |  | ||||||
|             const option = options[key]; |  | ||||||
|             const validators = option.validators; |             const validators = option.validators; | ||||||
|             const validation_response = apply_validators(validators, [ |             for (const validator of validators) { | ||||||
|                 value, |                 const response = validator(value, raw); | ||||||
|                 value, |                 if (response !== true) { | ||||||
|             ]); |                     logger.error( | ||||||
|             if (validation_response !== true) { |                         `Failed to validate overridden option: ${key}`, | ||||||
|                 return `In overridden option ${key} for ${language}, ${validation_response}`; |                         response | ||||||
|  |                     ); | ||||||
|  |                     return false; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |             overrides[language][key] = value; | ||||||
|  |         } | ||||||
|  |         // Modifies the reference
 | ||||||
|  |         options[ | ||||||
|  |             options.index_of(options.find(o => o.key === 'limit_overrides')) | ||||||
|  |         ] = overrides; | ||||||
|     } |     } | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const options = [ | ||||||
|  |     { | ||||||
|  |         key: 'log_level', | ||||||
|  |         desc: 'Level of data to log', | ||||||
|  |         default: 'INFO', | ||||||
|  |         options: Object.values(Logger.LogLevels), | ||||||
|  |         validators: [ | ||||||
|  |             x => | ||||||
|  |                 Object.values(Logger.LogLevels).includes(x) || | ||||||
|  |                 `Log level ${x} does not exist`, | ||||||
|  |         ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'bind_address', | ||||||
|  |         desc: 'Address to bind REST API on', | ||||||
|  |         default: `0.0.0.0:${process.env["PORT"] || 2000}`, | ||||||
|  |         validators: [], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'data_directory', | ||||||
|  |         desc: 'Absolute path to store all piston related data at', | ||||||
|  |         default: '/piston', | ||||||
|  |         validators: [ | ||||||
|  |             x => fss.exists_sync(x) || `Directory ${x} does not exist`, | ||||||
|  |         ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'runner_uid_min', | ||||||
|  |         desc: 'Minimum uid to use for runner', | ||||||
|  |         default: 1001, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'runner_uid_max', | ||||||
|  |         desc: 'Maximum uid to use for runner', | ||||||
|  |         default: 1500, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'runner_gid_min', | ||||||
|  |         desc: 'Minimum gid to use for runner', | ||||||
|  |         default: 1001, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'runner_gid_max', | ||||||
|  |         desc: 'Maximum gid to use for runner', | ||||||
|  |         default: 1500, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'disable_networking', | ||||||
|  |         desc: 'Set to true to disable networking', | ||||||
|  |         default: true, | ||||||
|  |         parser: x => x === 'true', | ||||||
|  |         validators: [x => typeof x === 'boolean' || `${x} is not a boolean`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'output_max_size', | ||||||
|  |         desc: 'Max size of each stdio buffer', | ||||||
|  |         default: 1024, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'max_process_count', | ||||||
|  |         desc: 'Max number of processes per job', | ||||||
|  |         default: 64, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'max_open_files', | ||||||
|  |         desc: 'Max number of open files per job', | ||||||
|  |         default: 2048, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'max_file_size', | ||||||
|  |         desc: 'Max file size in bytes for a file', | ||||||
|  |         default: 10000000, //10MB
 | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'compile_timeout', | ||||||
|  |         desc: 'Max time allowed for compile stage in milliseconds', | ||||||
|  |         default: 10000, // 10 seconds
 | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'run_timeout', | ||||||
|  |         desc: 'Max time allowed for run stage in milliseconds', | ||||||
|  |         default: 3000, // 3 seconds
 | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'compile_memory_limit', | ||||||
|  |         desc: 'Max memory usage for compile stage in bytes (set to -1 for no limit)', | ||||||
|  |         default: -1, // no limit
 | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'run_memory_limit', | ||||||
|  |         desc: 'Max memory usage for run stage in bytes (set to -1 for no limit)', | ||||||
|  |         default: -1, // no limit
 | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [(x, raw) => !is_nan(x) || `${raw} is not a number`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'flake_path', | ||||||
|  |         desc: 'Path to nix flake defining runtimes to install', | ||||||
|  |         default: 'github:engineer-man/piston?directory=packages', | ||||||
|  |         validators: [], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'runtime_set', | ||||||
|  |         desc: 'Key on the flake specified by flake_path to access runtimes from', | ||||||
|  |         default: 'all', | ||||||
|  |         validators: [] | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'max_concurrent_jobs', | ||||||
|  |         desc: 'Maximum number of concurrent jobs to run at one time', | ||||||
|  |         default: 64, | ||||||
|  |         parser: parse_int, | ||||||
|  |         validators: [x => x > 0 || `${x} cannot be negative`], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         key: 'limit_overrides', | ||||||
|  |         desc: 'Per-language exceptions in JSON format for each of:\ | ||||||
|  |         max_process_count, max_open_files, max_file_size, compile_memory_limit,\ | ||||||
|  |         run_memory_limit, compile_timeout, run_timeout, output_max_size', | ||||||
|  |         default: {}, | ||||||
|  |         parser: parse_overrides, | ||||||
|  |         validators: [ | ||||||
|  |             x => !!x || `Invalid JSON format for the overrides\n${x}`, | ||||||
|  |             (overrides, _, options) => | ||||||
|  |                 validate_overrides(overrides, options) || | ||||||
|  |                 `Failed to validate the overrides`, | ||||||
|  |         ], | ||||||
|  |     }, | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
| logger.info(`Loading Configuration from environment`); | logger.info(`Loading Configuration from environment`); | ||||||
| 
 | 
 | ||||||
|  | let errored = false; | ||||||
|  | 
 | ||||||
| let config = {}; | let config = {}; | ||||||
| 
 | 
 | ||||||
| for (const option_name in options) { | options.forEach(option => { | ||||||
|     const env_key = 'PISTON_' + option_name.to_upper_case(); |     const env_key = 'PISTON_' + option.key.to_upper_case(); | ||||||
|     const option = options[option_name]; | 
 | ||||||
|     const parser = option.parser || (x => x); |     const parser = option.parser || (x => x); | ||||||
|  | 
 | ||||||
|     const env_val = process.env[env_key]; |     const env_val = process.env[env_key]; | ||||||
|  | 
 | ||||||
|     const parsed_val = parser(env_val); |     const parsed_val = parser(env_val); | ||||||
|  | 
 | ||||||
|     const value = env_val === undefined ? option.default : parsed_val; |     const value = env_val === undefined ? option.default : parsed_val; | ||||||
|     const validator_parameters = | 
 | ||||||
|         env_val === undefined ? [value, value] : [parsed_val, env_val]; |     option.validators.for_each(validator => { | ||||||
|     const validation_response = apply_validators( |         let response = null; | ||||||
|         option.validators, |         if (env_val) response = validator(parsed_val, env_val, options); | ||||||
|         validator_parameters |         else response = validator(value, value, options); | ||||||
|     ); | 
 | ||||||
|     if (validation_response !== true) { |         if (response !== true) { | ||||||
|  |             errored = true; | ||||||
|             logger.error( |             logger.error( | ||||||
|             `Config option ${option_name} failed validation:`, |                 `Config option ${option.key} failed validation:`, | ||||||
|             validation_response |                 response | ||||||
|             ); |             ); | ||||||
|         process.exit(1); |             return; | ||||||
|         } |         } | ||||||
|     config[option_name] = value; |     }); | ||||||
|  | 
 | ||||||
|  |     config[option.key] = value; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | if (errored) { | ||||||
|  |     process.exit(1); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| logger.info('Configuration successfully loaded'); | logger.info('Configuration successfully loaded'); | ||||||
|  |  | ||||||
|  | @ -1,6 +1,9 @@ | ||||||
| const logger = require('logplease').create('runtime'); | const logger = require('logplease').create('runtime'); | ||||||
| const cp = require('child_process'); | const cp = require('child_process'); | ||||||
| const config = require('./config'); | const config = require('./config'); | ||||||
|  | const globals = require('./globals'); | ||||||
|  | const fss = require('fs'); | ||||||
|  | const path = require('path'); | ||||||
| 
 | 
 | ||||||
| const runtimes = []; | const runtimes = []; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										9
									
								
								packages/MATL/22.5.0/build.sh
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								packages/MATL/22.5.0/build.sh
									
										
									
									
										vendored
									
									
								
							|  | @ -1,9 +0,0 @@ | ||||||
| #!/usr/bin/env bash |  | ||||||
| 
 |  | ||||||
| # build octave as dependency |  | ||||||
| source ../../octave/6.2.0/build.sh |  | ||||||
| 
 |  | ||||||
| # curl MATL 22.5.0 |  | ||||||
| curl -L "https://github.com/lmendo/MATL/archive/refs/tags/22.5.0.tar.gz" -o MATL.tar.xz |  | ||||||
| tar xf MATL.tar.xz --strip-components=1 |  | ||||||
| rm MATL.tar.xz |  | ||||||
							
								
								
									
										5
									
								
								packages/MATL/22.5.0/environment
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								packages/MATL/22.5.0/environment
									
										
									
									
										vendored
									
									
								
							|  | @ -1,5 +0,0 @@ | ||||||
| #!/usr/bin/env bash |  | ||||||
| 
 |  | ||||||
| # Path to MATL binary |  | ||||||
| export PATH=$PWD/bin:$PATH |  | ||||||
| export MATL_PATH=$PWD |  | ||||||
							
								
								
									
										5
									
								
								packages/MATL/22.5.0/metadata.json
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								packages/MATL/22.5.0/metadata.json
									
										
									
									
										vendored
									
									
								
							|  | @ -1,5 +0,0 @@ | ||||||
| { |  | ||||||
|     "language": "matl", |  | ||||||
|     "version": "22.5.0", |  | ||||||
|     "aliases": [] |  | ||||||
| } |  | ||||||
							
								
								
									
										13
									
								
								packages/MATL/22.5.0/run
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								packages/MATL/22.5.0/run
									
										
									
									
										vendored
									
									
								
							|  | @ -1,13 +0,0 @@ | ||||||
| #!/usr/bin/env bash |  | ||||||
| 
 |  | ||||||
| # get file as first argument |  | ||||||
| file="$1" |  | ||||||
| 
 |  | ||||||
| # remove the file from $@ |  | ||||||
| shift |  | ||||||
| 
 |  | ||||||
| # use the rest of the arguments as stdin |  | ||||||
| stdin=`printf "%s\n" "$@"` |  | ||||||
| 
 |  | ||||||
| # pass stdin into octave which will run MATL |  | ||||||
| echo "$stdin" | octave -W -p "$MATL_PATH" --eval "matl -of '$file'" |  | ||||||
							
								
								
									
										1
									
								
								packages/MATL/22.5.0/test.matl
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								packages/MATL/22.5.0/test.matl
									
										
									
									
										vendored
									
									
								
							|  | @ -1 +0,0 @@ | ||||||
| 'OK' |  | ||||||
|  | @ -350,7 +350,6 @@ Content-Type: application/json | ||||||
| `llvm_ir`, | `llvm_ir`, | ||||||
| `lolcode`, | `lolcode`, | ||||||
| `lua`, | `lua`, | ||||||
| `matl`, |  | ||||||
| `nasm`, | `nasm`, | ||||||
| `nasm64`, | `nasm64`, | ||||||
| `nim`, | `nim`, | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue