| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 |
- #
- # Copyright 2025 The InfiniFlow Authors. All Rights Reserved.
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- #
- import asyncio
- import base64
- import json
- import os
- import time
- import uuid
-
- from core.config import TIMEOUT
- from core.container import allocate_container_blocking, release_container
- from core.logger import logger
- from models.enums import ResourceLimitType, ResultStatus, RuntimeErrorType, SupportLanguage, UnauthorizedAccessType
- from models.schemas import CodeExecutionRequest, CodeExecutionResult
- from utils.common import async_run_command
-
-
- async def execute_code(req: CodeExecutionRequest):
- """Fully asynchronous execution logic"""
- language = req.language
- container = await allocate_container_blocking(language)
- if not container:
- return CodeExecutionResult(
- status=ResultStatus.PROGRAM_RUNNER_ERROR,
- stdout="",
- stderr="Container pool is busy",
- exit_code=-10,
- detail="no_available_container",
- )
-
- task_id = str(uuid.uuid4())
- workdir = f"/tmp/sandbox_{task_id}"
- os.makedirs(workdir, mode=0o700, exist_ok=True)
-
- try:
- if language == SupportLanguage.PYTHON:
- code_name = "main.py"
- # code
- code_path = os.path.join(workdir, code_name)
- with open(code_path, "wb") as f:
- f.write(base64.b64decode(req.code_b64))
- # runner
- runner_name = "runner.py"
- runner_path = os.path.join(workdir, runner_name)
- with open(runner_path, "w") as f:
- f.write("""import json
- import os
- import sys
- sys.path.insert(0, os.path.dirname(__file__))
- from main import main
- if __name__ == "__main__":
- args = json.loads(sys.argv[1])
- result = main(**args)
- if result is not None:
- print(result)
- """)
-
- elif language == SupportLanguage.NODEJS:
- code_name = "main.js"
- code_path = os.path.join(workdir, "main.js")
- with open(code_path, "wb") as f:
- f.write(base64.b64decode(req.code_b64))
-
- runner_name = "runner.js"
- runner_path = os.path.join(workdir, "runner.js")
- with open(runner_path, "w") as f:
- f.write("""
- const fs = require('fs');
- const path = require('path');
-
- const args = JSON.parse(process.argv[2]);
- const mainPath = path.join(__dirname, 'main.js');
-
- function isPromise(value) {
- return Boolean(value && typeof value.then === 'function');
- }
-
- if (fs.existsSync(mainPath)) {
- const mod = require(mainPath);
- const main = typeof mod === 'function' ? mod : mod.main;
-
- if (typeof main !== 'function') {
- console.error('Error: main is not a function');
- process.exit(1);
- }
-
- if (typeof args === 'object' && args !== null) {
- try {
- const result = main(args);
- if (isPromise(result)) {
- result.then(output => {
- if (output !== null) {
- console.log(output);
- }
- }).catch(err => {
- console.error('Error in async main function:', err);
- });
- } else {
- if (result !== null) {
- console.log(result);
- }
- }
- } catch (err) {
- console.error('Error when executing main:', err);
- }
- } else {
- console.error('Error: args is not a valid object:', args);
- }
- } else {
- console.error('main.js not found in the current directory');
- }
- """)
- # dirs
- returncode, _, stderr = await async_run_command("docker", "exec", container, "mkdir", "-p", f"/workspace/{task_id}", timeout=5)
- if returncode != 0:
- raise RuntimeError(f"Directory creation failed: {stderr}")
-
- # archive
- tar_proc = await asyncio.create_subprocess_exec("tar", "czf", "-", "-C", workdir, code_name, runner_name, stdout=asyncio.subprocess.PIPE)
- tar_stdout, _ = await tar_proc.communicate()
-
- # unarchive
- docker_proc = await asyncio.create_subprocess_exec(
- "docker", "exec", "-i", container, "tar", "xzf", "-", "-C", f"/workspace/{task_id}", stdin=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE
- )
- stdout, stderr = await docker_proc.communicate(input=tar_stdout)
-
- if docker_proc.returncode != 0:
- raise RuntimeError(stderr.decode())
-
- # exec
- start_time = time.time()
- try:
- logger.info(f"Passed in args: {req.arguments}")
- args_json = json.dumps(req.arguments or {})
- run_args = [
- "docker",
- "exec",
- "--workdir",
- f"/workspace/{task_id}",
- container,
- "timeout",
- str(TIMEOUT),
- language,
- ]
- # flags
- if language == SupportLanguage.PYTHON:
- run_args.extend(["-I", "-B"])
- elif language == SupportLanguage.NODEJS:
- run_args.extend([])
- else:
- assert True, "Will never reach here"
- run_args.extend([runner_name, args_json])
-
- returncode, stdout, stderr = await async_run_command(
- *run_args,
- timeout=TIMEOUT + 5,
- )
-
- time_used_ms = (time.time() - start_time) * 1000
-
- logger.info("----------------------------------------------")
- logger.info(f"Code: {str(base64.b64decode(req.code_b64))}")
- logger.info(f"{returncode=}")
- logger.info(f"{stdout=}")
- logger.info(f"{stderr=}")
- logger.info(f"{args_json=}")
-
- if returncode == 0:
- return CodeExecutionResult(
- status=ResultStatus.SUCCESS,
- stdout=str(stdout),
- stderr=stderr,
- exit_code=0,
- time_used_ms=time_used_ms,
- )
- elif returncode == 124:
- return CodeExecutionResult(
- status=ResultStatus.RESOURCE_LIMIT_EXCEEDED,
- stdout="",
- stderr="Execution timeout",
- exit_code=-124,
- resource_limit_type=ResourceLimitType.TIME,
- time_used_ms=time_used_ms,
- )
- elif returncode == 137:
- return CodeExecutionResult(
- status=ResultStatus.RESOURCE_LIMIT_EXCEEDED,
- stdout="",
- stderr="Memory limit exceeded (killed by OOM)",
- exit_code=-137,
- resource_limit_type=ResourceLimitType.MEMORY,
- time_used_ms=time_used_ms,
- )
- return analyze_error_result(stderr, returncode)
-
- except asyncio.TimeoutError:
- await async_run_command("docker", "exec", container, "pkill", "-9", language)
- return CodeExecutionResult(
- status=ResultStatus.RESOURCE_LIMIT_EXCEEDED,
- stdout="",
- stderr="Execution timeout",
- exit_code=-1,
- resource_limit_type=ResourceLimitType.TIME,
- time_used_ms=(time.time() - start_time) * 1000,
- )
-
- except Exception as e:
- logger.error(f"Execution exception: {str(e)}")
- return CodeExecutionResult(status=ResultStatus.PROGRAM_RUNNER_ERROR, stdout="", stderr=str(e), exit_code=-3, detail="internal_error")
-
- finally:
- # cleanup
- cleanup_tasks = [async_run_command("docker", "exec", container, "rm", "-rf", f"/workspace/{task_id}"), async_run_command("rm", "-rf", workdir)]
- await asyncio.gather(*cleanup_tasks, return_exceptions=True)
- await release_container(container, language)
-
-
- def analyze_error_result(stderr: str, exit_code: int) -> CodeExecutionResult:
- """Analyze the error result and classify it"""
- if "Permission denied" in stderr:
- return CodeExecutionResult(
- status=ResultStatus.UNAUTHORIZED_ACCESS,
- stdout="",
- stderr=stderr,
- exit_code=exit_code,
- unauthorized_access_type=UnauthorizedAccessType.FILE_ACCESS,
- )
- elif "Operation not permitted" in stderr:
- return CodeExecutionResult(
- status=ResultStatus.UNAUTHORIZED_ACCESS,
- stdout="",
- stderr=stderr,
- exit_code=exit_code,
- unauthorized_access_type=UnauthorizedAccessType.DISALLOWED_SYSCALL,
- )
- elif "MemoryError" in stderr:
- return CodeExecutionResult(
- status=ResultStatus.RESOURCE_LIMIT_EXCEEDED,
- stdout="",
- stderr=stderr,
- exit_code=exit_code,
- resource_limit_type=ResourceLimitType.MEMORY,
- )
- else:
- return CodeExecutionResult(
- status=ResultStatus.PROGRAM_ERROR,
- stdout="",
- stderr=stderr,
- exit_code=exit_code,
- runtime_error_type=RuntimeErrorType.NONZERO_EXIT,
- )
|